Merge remote-tracking branch 'misskey/develop' into future-2024-04-25
This commit is contained in:
		
						commit
						4fe8a26081
					
				
					 58 changed files with 1305 additions and 306 deletions
				
			
		
							
								
								
									
										19
									
								
								CHANGELOG.md
									
										
									
									
									
								
							
							
						
						
									
										19
									
								
								CHANGELOG.md
									
										
									
									
									
								
							| 
						 | 
				
			
			@ -7,9 +7,18 @@
 | 
			
		|||
- Enhance: URLプレビューの有効化・無効化を設定できるように #13569
 | 
			
		||||
- Enhance: アンテナでBotによるノートを除外できるように  
 | 
			
		||||
  (Cherry-picked from https://github.com/MisskeyIO/misskey/pull/545)
 | 
			
		||||
- Enhance: クリップのノート数を表示するように
 | 
			
		||||
- Enhance: コンディショナルロールの条件として以下を新たに追加 (#13667)
 | 
			
		||||
  - 猫ユーザーか
 | 
			
		||||
  - botユーザーか
 | 
			
		||||
  - サスペンド済みユーザーか
 | 
			
		||||
  - 鍵アカウントユーザーか
 | 
			
		||||
  - 「アカウントを見つけやすくする」が有効なユーザーか
 | 
			
		||||
- Fix: Play作成時に設定した公開範囲が機能していない問題を修正
 | 
			
		||||
- Fix: 正規化されていない状態のhashtagが連合されてきたhtmlに含まれているとhashtagが正しくhashtagに復元されない問題を修正
 | 
			
		||||
 | 
			
		||||
### Client
 | 
			
		||||
- Feat: アップロードするファイルの名前をランダム文字列にできるように
 | 
			
		||||
- Enhance: 自分のノートの添付ファイルから直接ファイルの詳細ページに飛べるように
 | 
			
		||||
- Enhance: 広告がMisskeyと同一ドメインの場合はRouterで遷移するように
 | 
			
		||||
- Enhance: リアクション・いいねの総数を表示するように
 | 
			
		||||
| 
						 | 
				
			
			@ -25,6 +34,8 @@
 | 
			
		|||
- Enhance: ノートについているリアクションの「もっと!」から、リアクションの一覧を表示できるように
 | 
			
		||||
- Enhance: リプライにて引用がある場合テキストが空でもノートできるように
 | 
			
		||||
  - 引用したいノートのURLをコピーしリプライ投稿画面にペーストして添付することで達成できます
 | 
			
		||||
- Enhance: フォローするかどうかの確認ダイアログを出せるように
 | 
			
		||||
- Chore: AiScriptを0.18.0にバージョンアップ
 | 
			
		||||
- Fix: 一部のページ内リンクが正しく動作しない問題を修正
 | 
			
		||||
- Fix: 周年の実績が閏年を考慮しない問題を修正
 | 
			
		||||
- Fix: ローカルURLのプレビューポップアップが左上に表示される
 | 
			
		||||
| 
						 | 
				
			
			@ -38,6 +49,7 @@
 | 
			
		|||
- Fix: CWのみの引用リノートが詳細ページで純粋なリノートとして誤って扱われてしまう問題を修正
 | 
			
		||||
- Fix: ノート詳細ページにおいてCW付き引用リノートのCWボタンのラベルに「引用」が含まれていない問題を修正
 | 
			
		||||
- Fix: ダイアログの入力で字数制限に違反していてもEnterキーが押せてしまう問題を修正
 | 
			
		||||
- Fix: ダイレクト投稿の宛先が保存されない問題を修正
 | 
			
		||||
 | 
			
		||||
### Server
 | 
			
		||||
- Enhance: エンドポイント`antennas/update`の必須項目を`antennaId`のみに
 | 
			
		||||
| 
						 | 
				
			
			@ -47,6 +59,13 @@
 | 
			
		|||
- Fix: エンドポイント`notes/translate`のエラーを改善
 | 
			
		||||
- Fix: CleanRemoteFilesProcessorService report progress from 100% (#13632)
 | 
			
		||||
- Fix: 一部の音声ファイルが映像ファイルとして扱われる問題を修正
 | 
			
		||||
- Fix: リプライのみの引用リノートと、CWのみの引用リノートが純粋なリノートとして誤って扱われてしまう問題を修正
 | 
			
		||||
- Fix: 登録にメール認証が必須になっている場合、登録されているメールアドレスを削除できないように  
 | 
			
		||||
  (Cherry-picked from https://github.com/MisskeyIO/misskey/pull/606)
 | 
			
		||||
- Fix: nginx経由で/files/にRangeリクエストされた場合に正しく応答できないのを修正
 | 
			
		||||
- Fix: 一部のタイムラインのストリーミングでインスタンスミュートが効かない問題を修正
 | 
			
		||||
- Fix: グローバルタイムラインで返信が表示されないことがある問題を修正
 | 
			
		||||
- Fix: リノートをミュートしたユーザの投稿のリノートがミュートされる問題を修正
 | 
			
		||||
 | 
			
		||||
## 2024.3.1
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -30,9 +30,13 @@ Cypress.Commands.add('visitHome', () => {
 | 
			
		|||
})
 | 
			
		||||
 | 
			
		||||
Cypress.Commands.add('resetState', () => {
 | 
			
		||||
	// iframe.contentWindow.indexedDB.deleteDatabase() がchromeのバグで使用できないため、indexedDBを無効化している。
 | 
			
		||||
	// see https://github.com/misskey-dev/misskey/issues/13605#issuecomment-2053652123
 | 
			
		||||
	/*
 | 
			
		||||
	cy.window().then(win => {
 | 
			
		||||
		win.indexedDB.deleteDatabase('keyval-store');
 | 
			
		||||
	});
 | 
			
		||||
	 */
 | 
			
		||||
	cy.request('POST', '/api/reset-db', {}).as('reset');
 | 
			
		||||
	cy.get('@reset').its('status').should('equal', 204);
 | 
			
		||||
	cy.reload(true);
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										36
									
								
								locales/index.d.ts
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										36
									
								
								locales/index.d.ts
									
										
									
									
										vendored
									
									
								
							| 
						 | 
				
			
			@ -5133,6 +5133,22 @@ export interface Locale extends ILocale {
 | 
			
		|||
     * 動画・音声の再生にブラウザのUIを使用する
 | 
			
		||||
     */
 | 
			
		||||
    "useNativeUIForVideoAudioPlayer": string;
 | 
			
		||||
    /**
 | 
			
		||||
     * オリジナルのファイル名を保持
 | 
			
		||||
     */
 | 
			
		||||
    "keepOriginalFilename": string;
 | 
			
		||||
    /**
 | 
			
		||||
     * この設定をオフにすると、アップロード時にファイル名が自動でランダム文字列に置き換えられます。
 | 
			
		||||
     */
 | 
			
		||||
    "keepOriginalFilenameDescription": string;
 | 
			
		||||
    /**
 | 
			
		||||
     * 説明文はありません
 | 
			
		||||
     */
 | 
			
		||||
    "noDescription": string;
 | 
			
		||||
    /**
 | 
			
		||||
     * フォローの際常に確認する
 | 
			
		||||
     */
 | 
			
		||||
    "alwaysConfirmFollow": string;
 | 
			
		||||
    "_bubbleGame": {
 | 
			
		||||
        /**
 | 
			
		||||
         * 遊び方
 | 
			
		||||
| 
						 | 
				
			
			@ -6781,6 +6797,26 @@ export interface Locale extends ILocale {
 | 
			
		|||
             * リモートユーザー
 | 
			
		||||
             */
 | 
			
		||||
            "isRemote": string;
 | 
			
		||||
            /**
 | 
			
		||||
             * 猫ユーザー
 | 
			
		||||
             */
 | 
			
		||||
            "isCat": string;
 | 
			
		||||
            /**
 | 
			
		||||
             * botユーザー
 | 
			
		||||
             */
 | 
			
		||||
            "isBot": string;
 | 
			
		||||
            /**
 | 
			
		||||
             * サスペンド済みユーザー
 | 
			
		||||
             */
 | 
			
		||||
            "isSuspended": string;
 | 
			
		||||
            /**
 | 
			
		||||
             * 鍵アカウントユーザー
 | 
			
		||||
             */
 | 
			
		||||
            "isLocked": string;
 | 
			
		||||
            /**
 | 
			
		||||
             * 「アカウントを見つけやすくする」が有効なユーザー
 | 
			
		||||
             */
 | 
			
		||||
            "isExplorable": string;
 | 
			
		||||
            /**
 | 
			
		||||
             * アカウント作成から~以内
 | 
			
		||||
             */
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1279,6 +1279,10 @@ useTotp: "ワンタイムパスワードを使う"
 | 
			
		|||
useBackupCode: "バックアップコードを使う"
 | 
			
		||||
launchApp: "アプリを起動"
 | 
			
		||||
useNativeUIForVideoAudioPlayer: "動画・音声の再生にブラウザのUIを使用する"
 | 
			
		||||
keepOriginalFilename: "オリジナルのファイル名を保持"
 | 
			
		||||
keepOriginalFilenameDescription: "この設定をオフにすると、アップロード時にファイル名が自動でランダム文字列に置き換えられます。"
 | 
			
		||||
noDescription: "説明文はありません"
 | 
			
		||||
alwaysConfirmFollow: "フォローの際常に確認する"
 | 
			
		||||
 | 
			
		||||
_bubbleGame:
 | 
			
		||||
  howToPlay: "遊び方"
 | 
			
		||||
| 
						 | 
				
			
			@ -1750,6 +1754,11 @@ _role:
 | 
			
		|||
    roleAssignedTo: "マニュアルロールにアサイン済み"
 | 
			
		||||
    isLocal: "ローカルユーザー"
 | 
			
		||||
    isRemote: "リモートユーザー"
 | 
			
		||||
    isCat: "猫ユーザー"
 | 
			
		||||
    isBot: "botユーザー"
 | 
			
		||||
    isSuspended: "サスペンド済みユーザー"
 | 
			
		||||
    isLocked: "鍵アカウントユーザー"
 | 
			
		||||
    isExplorable: "「アカウントを見つけやすくする」が有効なユーザー"
 | 
			
		||||
    createdLessThan: "アカウント作成から~以内"
 | 
			
		||||
    createdMoreThan: "アカウント作成から~経過"
 | 
			
		||||
    followersLessThanOrEq: "フォロワー数が~以下"
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -19,6 +19,6 @@
 | 
			
		|||
	</head>
 | 
			
		||||
	<body>
 | 
			
		||||
		<redoc spec-url="/api.json" expand-responses="200" expand-single-schema-field="true"></redoc>
 | 
			
		||||
		<script src="https://cdn.redoc.ly/redoc/latest/bundles/redoc.standalone.js"></script>
 | 
			
		||||
		<script src="https://cdn.redoc.ly/redoc/v2.1.3/bundles/redoc.standalone.js" integrity="sha256-u4DgqzYXoArvNF/Ymw3puKexfOC6lYfw0sfmeliBJ1I=" crossorigin="anonymous"></script>
 | 
			
		||||
	</body>
 | 
			
		||||
</html>
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -13,7 +13,7 @@ import type { NotesRepository } from '@/models/_.js';
 | 
			
		|||
import { NoteEntityService } from '@/core/entities/NoteEntityService.js';
 | 
			
		||||
import { FanoutTimelineName, FanoutTimelineService } from '@/core/FanoutTimelineService.js';
 | 
			
		||||
import { isUserRelated } from '@/misc/is-user-related.js';
 | 
			
		||||
import { isPureRenote } from '@/misc/is-pure-renote.js';
 | 
			
		||||
import { isQuote, isRenote } from '@/misc/is-renote.js';
 | 
			
		||||
import { CacheService } from '@/core/CacheService.js';
 | 
			
		||||
import { isReply } from '@/misc/is-reply.js';
 | 
			
		||||
import { isInstanceMuted } from '@/misc/is-instance-muted.js';
 | 
			
		||||
| 
						 | 
				
			
			@ -101,7 +101,7 @@ export class FanoutTimelineEndpointService {
 | 
			
		|||
 | 
			
		||||
			if (ps.excludePureRenotes) {
 | 
			
		||||
				const parentFilter = filter;
 | 
			
		||||
				filter = (note) => !isPureRenote(note) && parentFilter(note);
 | 
			
		||||
				filter = (note) => (!isRenote(note) || isQuote(note)) && parentFilter(note);
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			if (ps.me) {
 | 
			
		||||
| 
						 | 
				
			
			@ -122,9 +122,7 @@ export class FanoutTimelineEndpointService {
 | 
			
		|||
				filter = (note) => {
 | 
			
		||||
					if (isUserRelated(note, userIdsWhoBlockingMe, ps.ignoreAuthorFromBlock)) return false;
 | 
			
		||||
					if (isUserRelated(note, userIdsWhoMeMuting, ps.ignoreAuthorFromMute)) return false;
 | 
			
		||||
					if (note.mentions.some(mention => userIdsWhoMeMuting.has(mention))) return false;
 | 
			
		||||
					if (isPureRenote(note) && note.renote && note.renote.mentions.some(mention => userIdsWhoMeMuting.has(mention))) return false;
 | 
			
		||||
					if (isPureRenote(note) && isUserRelated(note, userIdsWhoMeMutingRenotes, ps.ignoreAuthorFromMute)) return false;
 | 
			
		||||
					if (!ps.ignoreAuthorFromMute && isRenote(note) && !isQuote(note) && userIdsWhoMeMutingRenotes.has(note.userId)) return false;
 | 
			
		||||
					if (isInstanceMuted(note, userMutedInstances)) return false;
 | 
			
		||||
 | 
			
		||||
					return parentFilter(note);
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -10,6 +10,7 @@ import { Window } from 'happy-dom';
 | 
			
		|||
import { DI } from '@/di-symbols.js';
 | 
			
		||||
import type { Config } from '@/config.js';
 | 
			
		||||
import { intersperse } from '@/misc/prelude/array.js';
 | 
			
		||||
import { normalizeForSearch } from '@/misc/normalize-for-search.js';
 | 
			
		||||
import type { IMentionedRemoteUsers } from '@/models/Note.js';
 | 
			
		||||
import { bindThis } from '@/decorators.js';
 | 
			
		||||
import * as TreeAdapter from '../../node_modules/parse5/dist/tree-adapters/default.js';
 | 
			
		||||
| 
						 | 
				
			
			@ -33,6 +34,8 @@ export class MfmService {
 | 
			
		|||
		// some AP servers like Pixelfed use br tags as well as newlines
 | 
			
		||||
		html = html.replace(/<br\s?\/?>\r?\n/gi, '\n');
 | 
			
		||||
 | 
			
		||||
		const normalizedHashtagNames = hashtagNames == null ? undefined : new Set<string>(hashtagNames.map(x => normalizeForSearch(x)));
 | 
			
		||||
 | 
			
		||||
		const dom = parse5.parseFragment(html);
 | 
			
		||||
 | 
			
		||||
		let text = '';
 | 
			
		||||
| 
						 | 
				
			
			@ -85,7 +88,7 @@ export class MfmService {
 | 
			
		|||
					const href = node.attrs.find(x => x.name === 'href');
 | 
			
		||||
 | 
			
		||||
					// ハッシュタグ
 | 
			
		||||
					if (hashtagNames && href && hashtagNames.map(x => x.toLowerCase()).includes(txt.toLowerCase())) {
 | 
			
		||||
					if (normalizedHashtagNames && href && normalizedHashtagNames.has(normalizeForSearch(txt))) {
 | 
			
		||||
						text += txt;
 | 
			
		||||
					// メンション
 | 
			
		||||
					} else if (txt.startsWith('@') && !(rel && rel.value.startsWith('me '))) {
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -495,7 +495,7 @@ export class NoteCreateService implements OnApplicationShutdown {
 | 
			
		|||
		}
 | 
			
		||||
 | 
			
		||||
		// Check blocking
 | 
			
		||||
		if (data.renote && !this.isQuote(data)) {
 | 
			
		||||
		if (this.isRenote(data) && !this.isQuote(data)) {
 | 
			
		||||
			if (data.renote.userHost === null) {
 | 
			
		||||
				if (data.renote.userId !== user.id) {
 | 
			
		||||
					const blocked = await this.userBlockingService.checkBlocked(data.renote.userId, user.id);
 | 
			
		||||
| 
						 | 
				
			
			@ -855,7 +855,7 @@ export class NoteCreateService implements OnApplicationShutdown {
 | 
			
		|||
			}
 | 
			
		||||
 | 
			
		||||
			// If it is renote
 | 
			
		||||
			if (data.renote) {
 | 
			
		||||
			if (this.isRenote(data)) {
 | 
			
		||||
				const type = this.isQuote(data) ? 'quote' : 'renote';
 | 
			
		||||
 | 
			
		||||
				// Notify
 | 
			
		||||
| 
						 | 
				
			
			@ -1055,9 +1055,20 @@ export class NoteCreateService implements OnApplicationShutdown {
 | 
			
		|||
	}
 | 
			
		||||
 | 
			
		||||
	@bindThis
 | 
			
		||||
	private isQuote(note: Option): note is Option & { renote: MiNote } {
 | 
			
		||||
		// sync with misc/is-quote.ts
 | 
			
		||||
		return !!note.renote && (!!note.text || !!note.cw || (!!note.files && !!note.files.length) || !!note.poll);
 | 
			
		||||
	private isRenote(note: Option): note is Option & { renote: MiNote } {
 | 
			
		||||
		return note.renote != null;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	@bindThis
 | 
			
		||||
	private isQuote(note: Option & { renote: MiNote }): note is Option & { renote: MiNote } & (
 | 
			
		||||
		{ text: string } | { cw: string } | { reply: MiNote } | { poll: IPoll } | { files: MiDriveFile[] }
 | 
			
		||||
	) {
 | 
			
		||||
		// NOTE: SYNC WITH misc/is-quote.ts
 | 
			
		||||
		return note.text != null ||
 | 
			
		||||
			note.reply != null ||
 | 
			
		||||
			note.cw != null ||
 | 
			
		||||
			note.poll != null ||
 | 
			
		||||
			(note.files != null && note.files.length > 0);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	@bindThis
 | 
			
		||||
| 
						 | 
				
			
			@ -1133,7 +1144,7 @@ export class NoteCreateService implements OnApplicationShutdown {
 | 
			
		|||
	private async renderNoteOrRenoteActivity(data: Option, note: MiNote) {
 | 
			
		||||
		if (data.localOnly) return null;
 | 
			
		||||
 | 
			
		||||
		const content = data.renote && !this.isQuote(data)
 | 
			
		||||
		const content = this.isRenote(data) && !this.isQuote(data)
 | 
			
		||||
			? this.apRendererService.renderAnnounce(data.renote.uri ? data.renote.uri : `${this.config.url}/notes/${data.renote.id}`, note)
 | 
			
		||||
			: this.apRendererService.renderCreate(await this.apRendererService.renderNote(note, false), note);
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -24,7 +24,7 @@ import { bindThis } from '@/decorators.js';
 | 
			
		|||
import { MetaService } from '@/core/MetaService.js';
 | 
			
		||||
import { SearchService } from '@/core/SearchService.js';
 | 
			
		||||
import { ModerationLogService } from '@/core/ModerationLogService.js';
 | 
			
		||||
import { isPureRenote } from '@/misc/is-pure-renote.js';
 | 
			
		||||
import { isQuote, isRenote } from '@/misc/is-renote.js';
 | 
			
		||||
 | 
			
		||||
@Injectable()
 | 
			
		||||
export class NoteDeleteService {
 | 
			
		||||
| 
						 | 
				
			
			@ -86,7 +86,7 @@ export class NoteDeleteService {
 | 
			
		|||
				let renote: MiNote | null = null;
 | 
			
		||||
 | 
			
		||||
				// if deleted note is renote
 | 
			
		||||
				if (isPureRenote(note)) {
 | 
			
		||||
				if (isRenote(note) && !isQuote(note)) {
 | 
			
		||||
					renote = await this.notesRepository.findOneBy({
 | 
			
		||||
						id: note.renoteId,
 | 
			
		||||
					});
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -209,45 +209,79 @@ export class RoleService implements OnApplicationShutdown, OnModuleInit {
 | 
			
		|||
	private evalCond(user: MiUser, roles: MiRole[], value: RoleCondFormulaValue): boolean {
 | 
			
		||||
		try {
 | 
			
		||||
			switch (value.type) {
 | 
			
		||||
				// ~かつ~
 | 
			
		||||
				case 'and': {
 | 
			
		||||
					return value.values.every(v => this.evalCond(user, roles, v));
 | 
			
		||||
				}
 | 
			
		||||
				// ~または~
 | 
			
		||||
				case 'or': {
 | 
			
		||||
					return value.values.some(v => this.evalCond(user, roles, v));
 | 
			
		||||
				}
 | 
			
		||||
				// ~ではない
 | 
			
		||||
				case 'not': {
 | 
			
		||||
					return !this.evalCond(user, roles, value.value);
 | 
			
		||||
				}
 | 
			
		||||
				// マニュアルロールがアサインされている
 | 
			
		||||
				case 'roleAssignedTo': {
 | 
			
		||||
					return roles.some(r => r.id === value.roleId);
 | 
			
		||||
				}
 | 
			
		||||
				// ローカルユーザのみ
 | 
			
		||||
				case 'isLocal': {
 | 
			
		||||
					return this.userEntityService.isLocalUser(user);
 | 
			
		||||
				}
 | 
			
		||||
				// リモートユーザのみ
 | 
			
		||||
				case 'isRemote': {
 | 
			
		||||
					return this.userEntityService.isRemoteUser(user);
 | 
			
		||||
				}
 | 
			
		||||
				// サスペンド済みユーザである
 | 
			
		||||
				case 'isSuspended': {
 | 
			
		||||
					return user.isSuspended;
 | 
			
		||||
				}
 | 
			
		||||
				// 鍵アカウントユーザである
 | 
			
		||||
				case 'isLocked': {
 | 
			
		||||
					return user.isLocked;
 | 
			
		||||
				}
 | 
			
		||||
				// botユーザである
 | 
			
		||||
				case 'isBot': {
 | 
			
		||||
					return user.isBot;
 | 
			
		||||
				}
 | 
			
		||||
				// 猫である
 | 
			
		||||
				case 'isCat': {
 | 
			
		||||
					return user.isCat;
 | 
			
		||||
				}
 | 
			
		||||
				// 「ユーザを見つけやすくする」が有効なアカウント
 | 
			
		||||
				case 'isExplorable': {
 | 
			
		||||
					return user.isExplorable;
 | 
			
		||||
				}
 | 
			
		||||
				// ユーザが作成されてから指定期間経過した
 | 
			
		||||
				case 'createdLessThan': {
 | 
			
		||||
					return this.idService.parse(user.id).date.getTime() > (Date.now() - (value.sec * 1000));
 | 
			
		||||
				}
 | 
			
		||||
				// ユーザが作成されてから指定期間経っていない
 | 
			
		||||
				case 'createdMoreThan': {
 | 
			
		||||
					return this.idService.parse(user.id).date.getTime() < (Date.now() - (value.sec * 1000));
 | 
			
		||||
				}
 | 
			
		||||
				// フォロワー数が指定値以下
 | 
			
		||||
				case 'followersLessThanOrEq': {
 | 
			
		||||
					return user.followersCount <= value.value;
 | 
			
		||||
				}
 | 
			
		||||
				// フォロワー数が指定値以上
 | 
			
		||||
				case 'followersMoreThanOrEq': {
 | 
			
		||||
					return user.followersCount >= value.value;
 | 
			
		||||
				}
 | 
			
		||||
				// フォロー数が指定値以下
 | 
			
		||||
				case 'followingLessThanOrEq': {
 | 
			
		||||
					return user.followingCount <= value.value;
 | 
			
		||||
				}
 | 
			
		||||
				// フォロー数が指定値以上
 | 
			
		||||
				case 'followingMoreThanOrEq': {
 | 
			
		||||
					return user.followingCount >= value.value;
 | 
			
		||||
				}
 | 
			
		||||
				// ノート数が指定値以下
 | 
			
		||||
				case 'notesLessThanOrEq': {
 | 
			
		||||
					return user.notesCount <= value.value;
 | 
			
		||||
				}
 | 
			
		||||
				// ノート数が指定値以上
 | 
			
		||||
				case 'notesMoreThanOrEq': {
 | 
			
		||||
					return user.notesCount >= value.value;
 | 
			
		||||
				}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -5,7 +5,7 @@
 | 
			
		|||
 | 
			
		||||
import { Inject, Injectable } from '@nestjs/common';
 | 
			
		||||
import { DI } from '@/di-symbols.js';
 | 
			
		||||
import type { ClipFavoritesRepository, ClipsRepository, MiUser } from '@/models/_.js';
 | 
			
		||||
import type { ClipNotesRepository, ClipFavoritesRepository, ClipsRepository, MiUser } from '@/models/_.js';
 | 
			
		||||
import { awaitAll } from '@/misc/prelude/await-all.js';
 | 
			
		||||
import type { Packed } from '@/misc/json-schema.js';
 | 
			
		||||
import type { } from '@/models/Blocking.js';
 | 
			
		||||
| 
						 | 
				
			
			@ -20,6 +20,9 @@ export class ClipEntityService {
 | 
			
		|||
		@Inject(DI.clipsRepository)
 | 
			
		||||
		private clipsRepository: ClipsRepository,
 | 
			
		||||
 | 
			
		||||
		@Inject(DI.clipNotesRepository)
 | 
			
		||||
		private clipNotesRepository: ClipNotesRepository,
 | 
			
		||||
 | 
			
		||||
		@Inject(DI.clipFavoritesRepository)
 | 
			
		||||
		private clipFavoritesRepository: ClipFavoritesRepository,
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -47,6 +50,7 @@ export class ClipEntityService {
 | 
			
		|||
			isPublic: clip.isPublic,
 | 
			
		||||
			favoritedCount: await this.clipFavoritesRepository.countBy({ clipId: clip.id }),
 | 
			
		||||
			isFavorited: meId ? await this.clipFavoritesRepository.exists({ where: { clipId: clip.id, userId: meId } }) : undefined,
 | 
			
		||||
			notesCount: meId ? await this.clipNotesRepository.countBy({ clipId: clip.id }) : undefined,
 | 
			
		||||
		});
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,15 +0,0 @@
 | 
			
		|||
/*
 | 
			
		||||
 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
			
		||||
 * SPDX-License-Identifier: AGPL-3.0-only
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
import type { MiNote } from '@/models/Note.js';
 | 
			
		||||
 | 
			
		||||
export function isPureRenote(note: MiNote): note is MiNote & { renoteId: NonNullable<MiNote['renoteId']> } {
 | 
			
		||||
	if (!note.renoteId) return false;
 | 
			
		||||
 | 
			
		||||
	if (note.text) return false; // it's quoted with text
 | 
			
		||||
	if (note.fileIds.length !== 0) return false; // it's quoted with files
 | 
			
		||||
	if (note.hasPoll) return false; // it's quoted with poll
 | 
			
		||||
	return true;
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			@ -1,12 +0,0 @@
 | 
			
		|||
/*
 | 
			
		||||
 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
			
		||||
 * SPDX-License-Identifier: AGPL-3.0-only
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
import type { MiNote } from '@/models/Note.js';
 | 
			
		||||
 | 
			
		||||
// eslint-disable-next-line import/no-default-export
 | 
			
		||||
export default function(note: MiNote): boolean {
 | 
			
		||||
	// sync with NoteCreateService.isQuote
 | 
			
		||||
	return note.renoteId != null && (note.text != null || note.hasPoll || (note.fileIds != null && note.fileIds.length > 0));
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										67
									
								
								packages/backend/src/misc/is-renote.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										67
									
								
								packages/backend/src/misc/is-renote.ts
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,67 @@
 | 
			
		|||
/*
 | 
			
		||||
 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
			
		||||
 * SPDX-License-Identifier: AGPL-3.0-only
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
import type { MiNote } from '@/models/Note.js';
 | 
			
		||||
import type { Packed } from '@/misc/json-schema.js';
 | 
			
		||||
 | 
			
		||||
type Renote =
 | 
			
		||||
	MiNote & {
 | 
			
		||||
		renoteId: NonNullable<MiNote['renoteId']>
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
type Quote =
 | 
			
		||||
	Renote & ({
 | 
			
		||||
		text: NonNullable<MiNote['text']>
 | 
			
		||||
	} | {
 | 
			
		||||
		cw: NonNullable<MiNote['cw']>
 | 
			
		||||
	} | {
 | 
			
		||||
		replyId: NonNullable<MiNote['replyId']>
 | 
			
		||||
		reply: NonNullable<MiNote['reply']>
 | 
			
		||||
	} | {
 | 
			
		||||
		hasPoll: true
 | 
			
		||||
	});
 | 
			
		||||
 | 
			
		||||
export function isRenote(note: MiNote): note is Renote {
 | 
			
		||||
	return note.renoteId != null;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export function isQuote(note: Renote): note is Quote {
 | 
			
		||||
	// NOTE: SYNC WITH NoteCreateService.isQuote
 | 
			
		||||
	return note.text != null ||
 | 
			
		||||
		note.cw != null ||
 | 
			
		||||
		note.replyId != null ||
 | 
			
		||||
		note.hasPoll ||
 | 
			
		||||
		note.fileIds.length > 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type PackedRenote =
 | 
			
		||||
	Packed<'Note'> & {
 | 
			
		||||
		renoteId: NonNullable<Packed<'Note'>['renoteId']>
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
type PackedQuote =
 | 
			
		||||
	PackedRenote & ({
 | 
			
		||||
		text: NonNullable<Packed<'Note'>['text']>
 | 
			
		||||
	} | {
 | 
			
		||||
		cw: NonNullable<Packed<'Note'>['cw']>
 | 
			
		||||
	} | {
 | 
			
		||||
		replyId: NonNullable<Packed<'Note'>['replyId']>
 | 
			
		||||
	} | {
 | 
			
		||||
		poll: NonNullable<Packed<'Note'>['poll']>
 | 
			
		||||
	} | {
 | 
			
		||||
		fileIds: NonNullable<Packed<'Note'>['fileIds']>
 | 
			
		||||
	});
 | 
			
		||||
 | 
			
		||||
export function isRenotePacked(note: Packed<'Note'>): note is PackedRenote {
 | 
			
		||||
	return note.renoteId != null;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export function isQuotePacked(note: PackedRenote): note is PackedQuote {
 | 
			
		||||
	return note.text != null ||
 | 
			
		||||
		note.cw != null ||
 | 
			
		||||
		note.replyId != null ||
 | 
			
		||||
		note.poll != null ||
 | 
			
		||||
		(note.fileIds != null && note.fileIds.length > 0);
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			@ -48,6 +48,7 @@ import {
 | 
			
		|||
	packedRoleCondFormulaValueCreatedSchema,
 | 
			
		||||
	packedRoleCondFormulaFollowersOrFollowingOrNotesSchema,
 | 
			
		||||
	packedRoleCondFormulaValueSchema,
 | 
			
		||||
	packedRoleCondFormulaValueUserSettingBooleanSchema,
 | 
			
		||||
} from '@/models/json-schema/role.js';
 | 
			
		||||
import { packedAdSchema } from '@/models/json-schema/ad.js';
 | 
			
		||||
import { packedReversiGameLiteSchema, packedReversiGameDetailedSchema } from '@/models/json-schema/reversi-game.js';
 | 
			
		||||
| 
						 | 
				
			
			@ -97,6 +98,7 @@ export const refs = {
 | 
			
		|||
	RoleCondFormulaLogics: packedRoleCondFormulaLogicsSchema,
 | 
			
		||||
	RoleCondFormulaValueNot: packedRoleCondFormulaValueNot,
 | 
			
		||||
	RoleCondFormulaValueIsLocalOrRemote: packedRoleCondFormulaValueIsLocalOrRemoteSchema,
 | 
			
		||||
	RoleCondFormulaValueUserSettingBooleanSchema: packedRoleCondFormulaValueUserSettingBooleanSchema,
 | 
			
		||||
	RoleCondFormulaValueAssignedRole: packedRoleCondFormulaValueAssignedRoleSchema,
 | 
			
		||||
	RoleCondFormulaValueCreated: packedRoleCondFormulaValueCreatedSchema,
 | 
			
		||||
	RoleCondFormulaFollowersOrFollowingOrNotes: packedRoleCondFormulaFollowersOrFollowingOrNotesSchema,
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -6,69 +6,149 @@
 | 
			
		|||
import { Entity, Column, PrimaryColumn } from 'typeorm';
 | 
			
		||||
import { id } from './util/id.js';
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * ~かつ~
 | 
			
		||||
 * 複数の条件を同時に満たす場合のみ成立とする
 | 
			
		||||
 */
 | 
			
		||||
type CondFormulaValueAnd = {
 | 
			
		||||
	type: 'and';
 | 
			
		||||
	values: RoleCondFormulaValue[];
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * ~または~
 | 
			
		||||
 * 複数の条件のうち、いずれかを満たす場合のみ成立とする
 | 
			
		||||
 */
 | 
			
		||||
type CondFormulaValueOr = {
 | 
			
		||||
	type: 'or';
 | 
			
		||||
	values: RoleCondFormulaValue[];
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * ~ではない
 | 
			
		||||
 * 条件を満たさない場合のみ成立とする
 | 
			
		||||
 */
 | 
			
		||||
type CondFormulaValueNot = {
 | 
			
		||||
	type: 'not';
 | 
			
		||||
	value: RoleCondFormulaValue;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * ローカルユーザーのみ成立とする
 | 
			
		||||
 */
 | 
			
		||||
type CondFormulaValueIsLocal = {
 | 
			
		||||
	type: 'isLocal';
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * リモートユーザーのみ成立とする
 | 
			
		||||
 */
 | 
			
		||||
type CondFormulaValueIsRemote = {
 | 
			
		||||
	type: 'isRemote';
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * 既に指定のマニュアルロールにアサインされている場合のみ成立とする
 | 
			
		||||
 */
 | 
			
		||||
type CondFormulaValueRoleAssignedTo = {
 | 
			
		||||
	type: 'roleAssignedTo';
 | 
			
		||||
	roleId: string;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * サスペンド済みアカウントの場合のみ成立とする
 | 
			
		||||
 */
 | 
			
		||||
type CondFormulaValueIsSuspended = {
 | 
			
		||||
	type: 'isSuspended';
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * 鍵アカウントの場合のみ成立とする
 | 
			
		||||
 */
 | 
			
		||||
type CondFormulaValueIsLocked = {
 | 
			
		||||
	type: 'isLocked';
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * botアカウントの場合のみ成立とする
 | 
			
		||||
 */
 | 
			
		||||
type CondFormulaValueIsBot = {
 | 
			
		||||
	type: 'isBot';
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * 猫アカウントの場合のみ成立とする
 | 
			
		||||
 */
 | 
			
		||||
type CondFormulaValueIsCat = {
 | 
			
		||||
	type: 'isCat';
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * 「ユーザを見つけやすくする」が有効なアカウントの場合のみ成立とする
 | 
			
		||||
 */
 | 
			
		||||
type CondFormulaValueIsExplorable = {
 | 
			
		||||
	type: 'isExplorable';
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * ユーザが作成されてから指定期間経過した場合のみ成立とする
 | 
			
		||||
 */
 | 
			
		||||
type CondFormulaValueCreatedLessThan = {
 | 
			
		||||
	type: 'createdLessThan';
 | 
			
		||||
	sec: number;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * ユーザが作成されてから指定期間経っていない場合のみ成立とする
 | 
			
		||||
 */
 | 
			
		||||
type CondFormulaValueCreatedMoreThan = {
 | 
			
		||||
	type: 'createdMoreThan';
 | 
			
		||||
	sec: number;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * フォロワー数が指定値以下の場合のみ成立とする
 | 
			
		||||
 */
 | 
			
		||||
type CondFormulaValueFollowersLessThanOrEq = {
 | 
			
		||||
	type: 'followersLessThanOrEq';
 | 
			
		||||
	value: number;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * フォロワー数が指定値以上の場合のみ成立とする
 | 
			
		||||
 */
 | 
			
		||||
type CondFormulaValueFollowersMoreThanOrEq = {
 | 
			
		||||
	type: 'followersMoreThanOrEq';
 | 
			
		||||
	value: number;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * フォロー数が指定値以下の場合のみ成立とする
 | 
			
		||||
 */
 | 
			
		||||
type CondFormulaValueFollowingLessThanOrEq = {
 | 
			
		||||
	type: 'followingLessThanOrEq';
 | 
			
		||||
	value: number;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * フォロー数が指定値以上の場合のみ成立とする
 | 
			
		||||
 */
 | 
			
		||||
type CondFormulaValueFollowingMoreThanOrEq = {
 | 
			
		||||
	type: 'followingMoreThanOrEq';
 | 
			
		||||
	value: number;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * 投稿数が指定値以下の場合のみ成立とする
 | 
			
		||||
 */
 | 
			
		||||
type CondFormulaValueNotesLessThanOrEq = {
 | 
			
		||||
	type: 'notesLessThanOrEq';
 | 
			
		||||
	value: number;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * 投稿数が指定値以上の場合のみ成立とする
 | 
			
		||||
 */
 | 
			
		||||
type CondFormulaValueNotesMoreThanOrEq = {
 | 
			
		||||
	type: 'notesMoreThanOrEq';
 | 
			
		||||
	value: number;
 | 
			
		||||
| 
						 | 
				
			
			@ -80,6 +160,11 @@ export type RoleCondFormulaValue = { id: string } & (
 | 
			
		|||
	CondFormulaValueNot |
 | 
			
		||||
	CondFormulaValueIsLocal |
 | 
			
		||||
	CondFormulaValueIsRemote |
 | 
			
		||||
	CondFormulaValueIsSuspended |
 | 
			
		||||
	CondFormulaValueIsLocked |
 | 
			
		||||
	CondFormulaValueIsBot |
 | 
			
		||||
	CondFormulaValueIsCat |
 | 
			
		||||
	CondFormulaValueIsExplorable |
 | 
			
		||||
	CondFormulaValueRoleAssignedTo |
 | 
			
		||||
	CondFormulaValueCreatedLessThan |
 | 
			
		||||
	CondFormulaValueCreatedMoreThan |
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -52,5 +52,9 @@ export const packedClipSchema = {
 | 
			
		|||
			type: 'boolean',
 | 
			
		||||
			optional: true, nullable: false,
 | 
			
		||||
		},
 | 
			
		||||
		notesCount: {
 | 
			
		||||
			type: 'integer',
 | 
			
		||||
			optional: true, nullable: false,
 | 
			
		||||
		},
 | 
			
		||||
	},
 | 
			
		||||
} as const;
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -57,6 +57,20 @@ export const packedRoleCondFormulaValueIsLocalOrRemoteSchema = {
 | 
			
		|||
	},
 | 
			
		||||
} as const;
 | 
			
		||||
 | 
			
		||||
export const packedRoleCondFormulaValueUserSettingBooleanSchema = {
 | 
			
		||||
	type: 'object',
 | 
			
		||||
	properties: {
 | 
			
		||||
		id: {
 | 
			
		||||
			type: 'string', optional: false,
 | 
			
		||||
		},
 | 
			
		||||
		type: {
 | 
			
		||||
			type: 'string',
 | 
			
		||||
			nullable: false, optional: false,
 | 
			
		||||
			enum: ['isSuspended', 'isLocked', 'isBot', 'isCat', 'isExplorable'],
 | 
			
		||||
		},
 | 
			
		||||
	},
 | 
			
		||||
} as const;
 | 
			
		||||
 | 
			
		||||
export const packedRoleCondFormulaValueAssignedRoleSchema = {
 | 
			
		||||
	type: 'object',
 | 
			
		||||
	properties: {
 | 
			
		||||
| 
						 | 
				
			
			@ -135,6 +149,9 @@ export const packedRoleCondFormulaValueSchema = {
 | 
			
		|||
		{
 | 
			
		||||
			ref: 'RoleCondFormulaValueIsLocalOrRemote',
 | 
			
		||||
		},
 | 
			
		||||
		{
 | 
			
		||||
			ref: 'RoleCondFormulaValueUserSettingBooleanSchema',
 | 
			
		||||
		},
 | 
			
		||||
		{
 | 
			
		||||
			ref: 'RoleCondFormulaValueAssignedRole',
 | 
			
		||||
		},
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -33,7 +33,7 @@ import { UtilityService } from '@/core/UtilityService.js';
 | 
			
		|||
import { UserEntityService } from '@/core/entities/UserEntityService.js';
 | 
			
		||||
import { bindThis } from '@/decorators.js';
 | 
			
		||||
import { IActivity } from '@/core/activitypub/type.js';
 | 
			
		||||
import { isPureRenote } from '@/misc/is-pure-renote.js';
 | 
			
		||||
import { isQuote, isRenote } from '@/misc/is-renote.js';
 | 
			
		||||
import type { FastifyInstance, FastifyRequest, FastifyReply, FastifyPluginOptions, FastifyBodyParser } from 'fastify';
 | 
			
		||||
import type { FindOptionsWhere } from 'typeorm';
 | 
			
		||||
import type Logger from '@/logger.js';
 | 
			
		||||
| 
						 | 
				
			
			@ -107,7 +107,7 @@ export class ActivityPubServerService {
 | 
			
		|||
	 */
 | 
			
		||||
	@bindThis
 | 
			
		||||
	private async packActivity(note: MiNote): Promise<any> {
 | 
			
		||||
		if (isPureRenote(note)) {
 | 
			
		||||
		if (isRenote(note) && !isQuote(note)) {
 | 
			
		||||
			const renote = await this.notesRepository.findOneByOrFail({ id: note.renoteId });
 | 
			
		||||
			return this.apRendererService.renderAnnounce(renote.uri ? renote.uri : `${this.config.url}/notes/${renote.id}`, note);
 | 
			
		||||
		}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -212,6 +212,8 @@ export class FileServerService {
 | 
			
		|||
				}
 | 
			
		||||
 | 
			
		||||
				reply.header('Content-Type', FILE_TYPE_BROWSERSAFE.includes(image.type) ? image.type : 'application/octet-stream');
 | 
			
		||||
				reply.header('Content-Length', file.file.size);
 | 
			
		||||
				reply.header('Cache-Control', 'max-age=31536000, immutable');
 | 
			
		||||
				reply.header('Content-Disposition',
 | 
			
		||||
					contentDisposition(
 | 
			
		||||
						'inline',
 | 
			
		||||
| 
						 | 
				
			
			@ -254,6 +256,7 @@ export class FileServerService {
 | 
			
		|||
				return fs.createReadStream(file.path);
 | 
			
		||||
			} else {
 | 
			
		||||
				reply.header('Content-Type', FILE_TYPE_BROWSERSAFE.includes(file.file.type) ? file.file.type : 'application/octet-stream');
 | 
			
		||||
				reply.header('Content-Length', file.file.size);
 | 
			
		||||
				reply.header('Cache-Control', 'max-age=31536000, immutable');
 | 
			
		||||
				reply.header('Content-Disposition', contentDisposition('inline', file.filename));
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -528,9 +531,7 @@ export class FileServerService {
 | 
			
		|||
		if (!file.storedInternal) {
 | 
			
		||||
			if (!(file.isLink && file.uri)) return '204';
 | 
			
		||||
			const result = await this.downloadAndDetectTypeFromUrl(file.uri);
 | 
			
		||||
			if (!file.size) {
 | 
			
		||||
				file.size = (await fs.promises.stat(result.path)).size;
 | 
			
		||||
			}
 | 
			
		||||
			file.size = (await fs.promises.stat(result.path)).size;	// DB file.sizeは正確とは限らないので
 | 
			
		||||
			return {
 | 
			
		||||
				...result,
 | 
			
		||||
				url: file.uri,
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -122,12 +122,20 @@ export class ServerService implements OnApplicationShutdown {
 | 
			
		|||
				return;
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			const name = path.split('@')[0].replace(/\.webp$/i, '');
 | 
			
		||||
			const host = path.split('@')[1]?.replace(/\.webp$/i, '');
 | 
			
		||||
			const emojiPath = path.replace(/\.webp$/i, '');
 | 
			
		||||
			const pathChunks = emojiPath.split('@');
 | 
			
		||||
 | 
			
		||||
			if (pathChunks.length > 2) {
 | 
			
		||||
				reply.code(400);
 | 
			
		||||
				return;
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			const name = pathChunks.shift();
 | 
			
		||||
			const host = pathChunks.pop();
 | 
			
		||||
 | 
			
		||||
			const emoji = await this.emojisRepository.findOneBy({
 | 
			
		||||
				// `@.` is the spec of ReactionService.decodeReaction
 | 
			
		||||
				host: (host == null || host === '.') ? IsNull() : host,
 | 
			
		||||
				host: (host === undefined || host === '.') ? IsNull() : host,
 | 
			
		||||
				name: name,
 | 
			
		||||
			});
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -16,6 +16,7 @@ import { DI } from '@/di-symbols.js';
 | 
			
		|||
import { GlobalEventService } from '@/core/GlobalEventService.js';
 | 
			
		||||
import { L_CHARS, secureRndstr } from '@/misc/secure-rndstr.js';
 | 
			
		||||
import { UserAuthService } from '@/core/UserAuthService.js';
 | 
			
		||||
import { MetaService } from '@/core/MetaService.js';
 | 
			
		||||
import { ApiError } from '../../error.js';
 | 
			
		||||
 | 
			
		||||
export const meta = {
 | 
			
		||||
| 
						 | 
				
			
			@ -40,6 +41,12 @@ export const meta = {
 | 
			
		|||
			code: 'UNAVAILABLE',
 | 
			
		||||
			id: 'a2defefb-f220-8849-0af6-17f816099323',
 | 
			
		||||
		},
 | 
			
		||||
 | 
			
		||||
		emailRequired: {
 | 
			
		||||
			message: 'Email address is required.',
 | 
			
		||||
			code: 'EMAIL_REQUIRED',
 | 
			
		||||
			id: '324c7a88-59f2-492f-903f-89134f93e47e',
 | 
			
		||||
		},
 | 
			
		||||
	},
 | 
			
		||||
 | 
			
		||||
	res: {
 | 
			
		||||
| 
						 | 
				
			
			@ -67,6 +74,7 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint-
 | 
			
		|||
		@Inject(DI.userProfilesRepository)
 | 
			
		||||
		private userProfilesRepository: UserProfilesRepository,
 | 
			
		||||
 | 
			
		||||
		private metaService: MetaService,
 | 
			
		||||
		private userEntityService: UserEntityService,
 | 
			
		||||
		private emailService: EmailService,
 | 
			
		||||
		private userAuthService: UserAuthService,
 | 
			
		||||
| 
						 | 
				
			
			@ -98,6 +106,8 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint-
 | 
			
		|||
				if (!res.available) {
 | 
			
		||||
					throw new ApiError(meta.errors.unavailable);
 | 
			
		||||
				}
 | 
			
		||||
			} else if ((await this.metaService.fetch()).emailRequiredForSignup) {
 | 
			
		||||
				throw new ApiError(meta.errors.emailRequired);
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			await this.userProfilesRepository.update(me.id, {
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -16,7 +16,7 @@ import { Endpoint } from '@/server/api/endpoint-base.js';
 | 
			
		|||
import { NoteEntityService } from '@/core/entities/NoteEntityService.js';
 | 
			
		||||
import { NoteCreateService } from '@/core/NoteCreateService.js';
 | 
			
		||||
import { DI } from '@/di-symbols.js';
 | 
			
		||||
import { isPureRenote } from '@/misc/is-pure-renote.js';
 | 
			
		||||
import { isQuote, isRenote } from '@/misc/is-renote.js';
 | 
			
		||||
import { MetaService } from '@/core/MetaService.js';
 | 
			
		||||
import { UtilityService } from '@/core/UtilityService.js';
 | 
			
		||||
import { IdentifiableError } from '@/misc/identifiable-error.js';
 | 
			
		||||
| 
						 | 
				
			
			@ -286,7 +286,7 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint-
 | 
			
		|||
 | 
			
		||||
				if (renote == null) {
 | 
			
		||||
					throw new ApiError(meta.errors.noSuchRenoteTarget);
 | 
			
		||||
				} else if (isPureRenote(renote)) {
 | 
			
		||||
				} else if (isRenote(renote) && !isQuote(renote)) {
 | 
			
		||||
					throw new ApiError(meta.errors.cannotReRenote);
 | 
			
		||||
				}
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -332,7 +332,7 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint-
 | 
			
		|||
 | 
			
		||||
				if (reply == null) {
 | 
			
		||||
					throw new ApiError(meta.errors.noSuchReplyTarget);
 | 
			
		||||
				} else if (isPureRenote(reply)) {
 | 
			
		||||
				} else if (isRenote(reply) && !isQuote(reply)) {
 | 
			
		||||
					throw new ApiError(meta.errors.cannotReplyToPureRenote);
 | 
			
		||||
				} else if (!await this.noteEntityService.isVisibleForMe(reply, me.id)) {
 | 
			
		||||
					throw new ApiError(meta.errors.cannotReplyToInvisibleNote);
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -4,6 +4,10 @@
 | 
			
		|||
 */
 | 
			
		||||
 | 
			
		||||
import { bindThis } from '@/decorators.js';
 | 
			
		||||
import { isInstanceMuted } from '@/misc/is-instance-muted.js';
 | 
			
		||||
import { isUserRelated } from '@/misc/is-user-related.js';
 | 
			
		||||
import { isRenotePacked, isQuotePacked } from '@/misc/is-renote.js';
 | 
			
		||||
import type { Packed } from '@/misc/json-schema.js';
 | 
			
		||||
import type Connection from './Connection.js';
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
| 
						 | 
				
			
			@ -54,6 +58,24 @@ export default abstract class Channel {
 | 
			
		|||
		return this.connection.subscriber;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/*
 | 
			
		||||
	 * ミュートとブロックされてるを処理する
 | 
			
		||||
	 */
 | 
			
		||||
	protected isNoteMutedOrBlocked(note: Packed<'Note'>): boolean {
 | 
			
		||||
		// 流れてきたNoteがインスタンスミュートしたインスタンスが関わる
 | 
			
		||||
		if (isInstanceMuted(note, new Set<string>(this.userProfile?.mutedInstances ?? []))) return true;
 | 
			
		||||
 | 
			
		||||
		// 流れてきたNoteがミュートしているユーザーが関わる
 | 
			
		||||
		if (isUserRelated(note, this.userIdsWhoMeMuting)) return true;
 | 
			
		||||
		// 流れてきたNoteがブロックされているユーザーが関わる
 | 
			
		||||
		if (isUserRelated(note, this.userIdsWhoBlockingMe)) return true;
 | 
			
		||||
 | 
			
		||||
		// 流れてきたNoteがリノートをミュートしてるユーザが行ったもの
 | 
			
		||||
		if (isRenotePacked(note) && !isQuotePacked(note) && this.userIdsWhoMeMutingRenotes.has(note.user.id)) return true;
 | 
			
		||||
 | 
			
		||||
		return false;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	constructor(id: string, connection: Connection) {
 | 
			
		||||
		this.id = id;
 | 
			
		||||
		this.connection = connection;
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -4,7 +4,6 @@
 | 
			
		|||
 */
 | 
			
		||||
 | 
			
		||||
import { Injectable } from '@nestjs/common';
 | 
			
		||||
import { isUserRelated } from '@/misc/is-user-related.js';
 | 
			
		||||
import { NoteEntityService } from '@/core/entities/NoteEntityService.js';
 | 
			
		||||
import { bindThis } from '@/decorators.js';
 | 
			
		||||
import type { GlobalEvents } from '@/core/GlobalEventService.js';
 | 
			
		||||
| 
						 | 
				
			
			@ -40,12 +39,7 @@ class AntennaChannel extends Channel {
 | 
			
		|||
		if (data.type === 'note') {
 | 
			
		||||
			const note = await this.noteEntityService.pack(data.body.id, this.user, { detail: true });
 | 
			
		||||
 | 
			
		||||
			// 流れてきたNoteがミュートしているユーザーが関わるものだったら無視する
 | 
			
		||||
			if (isUserRelated(note, this.userIdsWhoMeMuting)) return;
 | 
			
		||||
			// 流れてきたNoteがブロックされているユーザーが関わるものだったら無視する
 | 
			
		||||
			if (isUserRelated(note, this.userIdsWhoBlockingMe)) return;
 | 
			
		||||
 | 
			
		||||
			if (note.renote && !note.text && isUserRelated(note, this.userIdsWhoMeMutingRenotes)) return;
 | 
			
		||||
			if (this.isNoteMutedOrBlocked(note)) return;
 | 
			
		||||
 | 
			
		||||
			this.connection.cacheNote(note);
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -4,10 +4,10 @@
 | 
			
		|||
 */
 | 
			
		||||
 | 
			
		||||
import { Injectable } from '@nestjs/common';
 | 
			
		||||
import { isUserRelated } from '@/misc/is-user-related.js';
 | 
			
		||||
import type { Packed } from '@/misc/json-schema.js';
 | 
			
		||||
import { NoteEntityService } from '@/core/entities/NoteEntityService.js';
 | 
			
		||||
import { bindThis } from '@/decorators.js';
 | 
			
		||||
import { isRenotePacked, isQuotePacked } from '@/misc/is-renote.js';
 | 
			
		||||
import Channel, { type MiChannelService } from '../channel.js';
 | 
			
		||||
 | 
			
		||||
class ChannelChannel extends Channel {
 | 
			
		||||
| 
						 | 
				
			
			@ -38,14 +38,9 @@ class ChannelChannel extends Channel {
 | 
			
		|||
	private async onNote(note: Packed<'Note'>) {
 | 
			
		||||
		if (note.channelId !== this.channelId) return;
 | 
			
		||||
 | 
			
		||||
		// 流れてきたNoteがミュートしているユーザーが関わるものだったら無視する
 | 
			
		||||
		if (isUserRelated(note, this.userIdsWhoMeMuting)) return;
 | 
			
		||||
		// 流れてきたNoteがブロックされているユーザーが関わるものだったら無視する
 | 
			
		||||
		if (isUserRelated(note, this.userIdsWhoBlockingMe)) return;
 | 
			
		||||
		if (this.isNoteMutedOrBlocked(note)) return;
 | 
			
		||||
 | 
			
		||||
		if (note.renote && !note.text && isUserRelated(note, this.userIdsWhoMeMutingRenotes)) return;
 | 
			
		||||
 | 
			
		||||
		if (this.user && note.renoteId && !note.text) {
 | 
			
		||||
		if (this.user && isRenotePacked(note) && !isQuotePacked(note)) {
 | 
			
		||||
			if (note.renote && Object.keys(note.renote.reactions).length > 0) {
 | 
			
		||||
				const myRenoteReaction = await this.noteEntityService.populateMyReaction(note.renote, this.user.id);
 | 
			
		||||
				note.renote.myReaction = myRenoteReaction;
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -4,14 +4,12 @@
 | 
			
		|||
 */
 | 
			
		||||
 | 
			
		||||
import { Injectable } from '@nestjs/common';
 | 
			
		||||
import { checkWordMute } from '@/misc/check-word-mute.js';
 | 
			
		||||
import { isInstanceMuted } from '@/misc/is-instance-muted.js';
 | 
			
		||||
import { isUserRelated } from '@/misc/is-user-related.js';
 | 
			
		||||
import type { Packed } from '@/misc/json-schema.js';
 | 
			
		||||
import { MetaService } from '@/core/MetaService.js';
 | 
			
		||||
import { NoteEntityService } from '@/core/entities/NoteEntityService.js';
 | 
			
		||||
import { bindThis } from '@/decorators.js';
 | 
			
		||||
import { RoleService } from '@/core/RoleService.js';
 | 
			
		||||
import { isRenotePacked, isQuotePacked } from '@/misc/is-renote.js';
 | 
			
		||||
import Channel, { type MiChannelService } from '../channel.js';
 | 
			
		||||
 | 
			
		||||
class GlobalTimelineChannel extends Channel {
 | 
			
		||||
| 
						 | 
				
			
			@ -55,31 +53,13 @@ class GlobalTimelineChannel extends Channel {
 | 
			
		|||
		if (note.visibility !== 'public') return;
 | 
			
		||||
		if (note.channelId != null) return;
 | 
			
		||||
 | 
			
		||||
		// 関係ない返信は除外
 | 
			
		||||
		if (note.reply && !this.following[note.userId]?.withReplies) {
 | 
			
		||||
			const reply = note.reply;
 | 
			
		||||
			// 「チャンネル接続主への返信」でもなければ、「チャンネル接続主が行った返信」でもなければ、「投稿者の投稿者自身への返信」でもない場合
 | 
			
		||||
			if (reply.userId !== this.user!.id && note.userId !== this.user!.id && reply.userId !== note.userId) return;
 | 
			
		||||
		}
 | 
			
		||||
		if (isRenotePacked(note) && !isQuotePacked(note) && !this.withRenotes) return;
 | 
			
		||||
 | 
			
		||||
		if (note.user.isSilenced && !this.following[note.userId] && note.userId !== this.user!.id) return;
 | 
			
		||||
 | 
			
		||||
		if (note.renote && note.text == null && (note.fileIds == null || note.fileIds.length === 0) && !this.withRenotes) return;
 | 
			
		||||
		if (this.isNoteMutedOrBlocked(note)) return;
 | 
			
		||||
 | 
			
		||||
		// Ignore notes from instances the user has muted
 | 
			
		||||
		if (isInstanceMuted(note, new Set<string>(this.userProfile?.mutedInstances ?? [])) && !this.following[note.userId]) return;
 | 
			
		||||
 | 
			
		||||
		// 流れてきたNoteがミュートしているユーザーが関わるものだったら無視する
 | 
			
		||||
		if (isUserRelated(note, this.userIdsWhoMeMuting)) return;
 | 
			
		||||
		// 流れてきたNoteがブロックされているユーザーが関わるものだったら無視する
 | 
			
		||||
		if (isUserRelated(note, this.userIdsWhoBlockingMe)) return;
 | 
			
		||||
 | 
			
		||||
		if (note.renote && !note.text && note.renote.mentions?.some(mention => this.userIdsWhoMeMuting.has(mention))) return;
 | 
			
		||||
		if (note.mentions?.some(mention => this.userIdsWhoMeMuting.has(mention))) return;
 | 
			
		||||
 | 
			
		||||
		if (note.renote && !note.text && isUserRelated(note, this.userIdsWhoMeMutingRenotes)) return;
 | 
			
		||||
 | 
			
		||||
		if (this.user && note.renoteId && !note.text) {
 | 
			
		||||
		if (this.user && isRenotePacked(note) && !isQuotePacked(note)) {
 | 
			
		||||
			if (note.renote && Object.keys(note.renote.reactions).length > 0) {
 | 
			
		||||
				const myRenoteReaction = await this.noteEntityService.populateMyReaction(note.renote, this.user.id);
 | 
			
		||||
				note.renote.myReaction = myRenoteReaction;
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -5,10 +5,10 @@
 | 
			
		|||
 | 
			
		||||
import { Injectable } from '@nestjs/common';
 | 
			
		||||
import { normalizeForSearch } from '@/misc/normalize-for-search.js';
 | 
			
		||||
import { isUserRelated } from '@/misc/is-user-related.js';
 | 
			
		||||
import type { Packed } from '@/misc/json-schema.js';
 | 
			
		||||
import { NoteEntityService } from '@/core/entities/NoteEntityService.js';
 | 
			
		||||
import { bindThis } from '@/decorators.js';
 | 
			
		||||
import { isRenotePacked, isQuotePacked } from '@/misc/is-renote.js';
 | 
			
		||||
import Channel, { type MiChannelService } from '../channel.js';
 | 
			
		||||
 | 
			
		||||
class HashtagChannel extends Channel {
 | 
			
		||||
| 
						 | 
				
			
			@ -43,14 +43,9 @@ class HashtagChannel extends Channel {
 | 
			
		|||
		const matched = this.q.some(tags => tags.every(tag => noteTags.includes(normalizeForSearch(tag))));
 | 
			
		||||
		if (!matched) return;
 | 
			
		||||
 | 
			
		||||
		// 流れてきたNoteがミュートしているユーザーが関わるものだったら無視する
 | 
			
		||||
		if (isUserRelated(note, this.userIdsWhoMeMuting)) return;
 | 
			
		||||
		// 流れてきたNoteがブロックされているユーザーが関わるものだったら無視する
 | 
			
		||||
		if (isUserRelated(note, this.userIdsWhoBlockingMe)) return;
 | 
			
		||||
		if (this.isNoteMutedOrBlocked(note)) return;
 | 
			
		||||
 | 
			
		||||
		if (note.renote && !note.text && isUserRelated(note, this.userIdsWhoMeMutingRenotes)) return;
 | 
			
		||||
 | 
			
		||||
		if (this.user && note.renoteId && !note.text) {
 | 
			
		||||
		if (this.user && isRenotePacked(note) && !isQuotePacked(note)) {
 | 
			
		||||
			if (note.renote && Object.keys(note.renote.reactions).length > 0) {
 | 
			
		||||
				const myRenoteReaction = await this.noteEntityService.populateMyReaction(note.renote, this.user.id);
 | 
			
		||||
				note.renote.myReaction = myRenoteReaction;
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -4,12 +4,10 @@
 | 
			
		|||
 */
 | 
			
		||||
 | 
			
		||||
import { Injectable } from '@nestjs/common';
 | 
			
		||||
import { checkWordMute } from '@/misc/check-word-mute.js';
 | 
			
		||||
import { isUserRelated } from '@/misc/is-user-related.js';
 | 
			
		||||
import { isInstanceMuted } from '@/misc/is-instance-muted.js';
 | 
			
		||||
import type { Packed } from '@/misc/json-schema.js';
 | 
			
		||||
import { NoteEntityService } from '@/core/entities/NoteEntityService.js';
 | 
			
		||||
import { bindThis } from '@/decorators.js';
 | 
			
		||||
import { isRenotePacked, isQuotePacked } from '@/misc/is-renote.js';
 | 
			
		||||
import Channel, { type MiChannelService } from '../channel.js';
 | 
			
		||||
 | 
			
		||||
class HomeTimelineChannel extends Channel {
 | 
			
		||||
| 
						 | 
				
			
			@ -51,9 +49,6 @@ class HomeTimelineChannel extends Channel {
 | 
			
		|||
			if (!isMe && !Object.hasOwn(this.following, note.userId)) return;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		// Ignore notes from instances the user has muted
 | 
			
		||||
		if (isInstanceMuted(note, new Set<string>(this.userProfile!.mutedInstances)) && !this.following[note.userId]) return;
 | 
			
		||||
 | 
			
		||||
		if (note.visibility === 'followers') {
 | 
			
		||||
			if (!isMe && !Object.hasOwn(this.following, note.userId)) return;
 | 
			
		||||
		} else if (note.visibility === 'specified') {
 | 
			
		||||
| 
						 | 
				
			
			@ -74,7 +69,7 @@ class HomeTimelineChannel extends Channel {
 | 
			
		|||
		if (note.user.isSilenced && !this.following[note.userId] && note.userId !== this.user!.id) return;
 | 
			
		||||
 | 
			
		||||
		// 純粋なリノート(引用リノートでないリノート)の場合
 | 
			
		||||
		if (note.renote && note.text == null && (note.fileIds == null || note.fileIds.length === 0) && note.poll == null) {
 | 
			
		||||
		if (isRenotePacked(note) && !isQuotePacked(note) && note.renote) {
 | 
			
		||||
			if (!this.withRenotes) return;
 | 
			
		||||
			if (note.renote.reply) {
 | 
			
		||||
				const reply = note.renote.reply;
 | 
			
		||||
| 
						 | 
				
			
			@ -83,17 +78,9 @@ class HomeTimelineChannel extends Channel {
 | 
			
		|||
			}
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		// 流れてきたNoteがミュートしているユーザーが関わるものだったら無視する
 | 
			
		||||
		if (isUserRelated(note, this.userIdsWhoMeMuting)) return;
 | 
			
		||||
		// 流れてきたNoteがブロックされているユーザーが関わるものだったら無視する
 | 
			
		||||
		if (isUserRelated(note, this.userIdsWhoBlockingMe)) return;
 | 
			
		||||
		if (this.isNoteMutedOrBlocked(note)) return;
 | 
			
		||||
 | 
			
		||||
		if (note.renote && !note.text && note.renote.mentions?.some(mention => this.userIdsWhoMeMuting.has(mention))) return;
 | 
			
		||||
		if (note.mentions?.some(mention => this.userIdsWhoMeMuting.has(mention))) return;
 | 
			
		||||
 | 
			
		||||
		if (note.renote && !note.text && isUserRelated(note, this.userIdsWhoMeMutingRenotes)) return;
 | 
			
		||||
 | 
			
		||||
		if (this.user && note.renoteId && !note.text) {
 | 
			
		||||
		if (this.user && isRenotePacked(note) && !isQuotePacked(note)) {
 | 
			
		||||
			if (note.renote && Object.keys(note.renote.reactions).length > 0) {
 | 
			
		||||
				const myRenoteReaction = await this.noteEntityService.populateMyReaction(note.renote, this.user.id);
 | 
			
		||||
				note.renote.myReaction = myRenoteReaction;
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -4,14 +4,12 @@
 | 
			
		|||
 */
 | 
			
		||||
 | 
			
		||||
import { Injectable } from '@nestjs/common';
 | 
			
		||||
import { checkWordMute } from '@/misc/check-word-mute.js';
 | 
			
		||||
import { isUserRelated } from '@/misc/is-user-related.js';
 | 
			
		||||
import { isInstanceMuted } from '@/misc/is-instance-muted.js';
 | 
			
		||||
import type { Packed } from '@/misc/json-schema.js';
 | 
			
		||||
import { MetaService } from '@/core/MetaService.js';
 | 
			
		||||
import { NoteEntityService } from '@/core/entities/NoteEntityService.js';
 | 
			
		||||
import { bindThis } from '@/decorators.js';
 | 
			
		||||
import { RoleService } from '@/core/RoleService.js';
 | 
			
		||||
import { isRenotePacked, isQuotePacked } from '@/misc/is-renote.js';
 | 
			
		||||
import Channel, { type MiChannelService } from '../channel.js';
 | 
			
		||||
 | 
			
		||||
class HybridTimelineChannel extends Channel {
 | 
			
		||||
| 
						 | 
				
			
			@ -74,8 +72,7 @@ class HybridTimelineChannel extends Channel {
 | 
			
		|||
			if (!isMe && !note.visibleUserIds!.includes(this.user!.id)) return;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		// Ignore notes from instances the user has muted
 | 
			
		||||
		if (isInstanceMuted(note, new Set<string>(this.userProfile!.mutedInstances)) && !this.following[note.userId]) return;
 | 
			
		||||
		if (this.isNoteMutedOrBlocked(note)) return;
 | 
			
		||||
 | 
			
		||||
		if (note.reply) {
 | 
			
		||||
			const reply = note.reply;
 | 
			
		||||
| 
						 | 
				
			
			@ -90,17 +87,7 @@ class HybridTimelineChannel extends Channel {
 | 
			
		|||
 | 
			
		||||
		if (note.user.isSilenced && !this.following[note.userId] && note.userId !== this.user!.id) return;
 | 
			
		||||
 | 
			
		||||
		if (note.renote && note.text == null && (note.fileIds == null || note.fileIds.length === 0) && !this.withRenotes) return;
 | 
			
		||||
 | 
			
		||||
		// 流れてきたNoteがミュートしているユーザーが関わるものだったら無視する
 | 
			
		||||
		if (isUserRelated(note, this.userIdsWhoMeMuting)) return;
 | 
			
		||||
		// 流れてきたNoteがブロックされているユーザーが関わるものだったら無視する
 | 
			
		||||
		if (isUserRelated(note, this.userIdsWhoBlockingMe)) return;
 | 
			
		||||
 | 
			
		||||
		if (note.renote && !note.text && note.renote.mentions?.some(mention => this.userIdsWhoMeMuting.has(mention))) return;
 | 
			
		||||
		if (note.mentions?.some(mention => this.userIdsWhoMeMuting.has(mention))) return;
 | 
			
		||||
 | 
			
		||||
		if (note.renote && !note.text && isUserRelated(note, this.userIdsWhoMeMutingRenotes)) return;
 | 
			
		||||
		if (isRenotePacked(note) && !isQuotePacked(note) && !this.withRenotes) return;
 | 
			
		||||
 | 
			
		||||
		if (this.user && note.renoteId && !note.text) {
 | 
			
		||||
			if (note.renote && Object.keys(note.renote.reactions).length > 0) {
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -4,13 +4,12 @@
 | 
			
		|||
 */
 | 
			
		||||
 | 
			
		||||
import { Injectable } from '@nestjs/common';
 | 
			
		||||
import { checkWordMute } from '@/misc/check-word-mute.js';
 | 
			
		||||
import { isUserRelated } from '@/misc/is-user-related.js';
 | 
			
		||||
import type { Packed } from '@/misc/json-schema.js';
 | 
			
		||||
import { MetaService } from '@/core/MetaService.js';
 | 
			
		||||
import { NoteEntityService } from '@/core/entities/NoteEntityService.js';
 | 
			
		||||
import { bindThis } from '@/decorators.js';
 | 
			
		||||
import { RoleService } from '@/core/RoleService.js';
 | 
			
		||||
import { isQuotePacked, isRenotePacked } from '@/misc/is-renote.js';
 | 
			
		||||
import Channel, { type MiChannelService } from '../channel.js';
 | 
			
		||||
 | 
			
		||||
class LocalTimelineChannel extends Channel {
 | 
			
		||||
| 
						 | 
				
			
			@ -66,19 +65,11 @@ class LocalTimelineChannel extends Channel {
 | 
			
		|||
 | 
			
		||||
		if (note.user.isSilenced && !this.following[note.userId] && note.userId !== this.user!.id) return;
 | 
			
		||||
 | 
			
		||||
		if (note.renote && note.text == null && (note.fileIds == null || note.fileIds.length === 0) && !this.withRenotes) return;
 | 
			
		||||
		if (isRenotePacked(note) && !isQuotePacked(note) && !this.withRenotes) return;
 | 
			
		||||
 | 
			
		||||
		// 流れてきたNoteがミュートしているユーザーが関わるものだったら無視する
 | 
			
		||||
		if (isUserRelated(note, this.userIdsWhoMeMuting)) return;
 | 
			
		||||
		// 流れてきたNoteがブロックされているユーザーが関わるものだったら無視する
 | 
			
		||||
		if (isUserRelated(note, this.userIdsWhoBlockingMe)) return;
 | 
			
		||||
		if (this.isNoteMutedOrBlocked(note)) return;
 | 
			
		||||
 | 
			
		||||
		if (note.renote && !note.text && note.renote.mentions?.some(mention => this.userIdsWhoMeMuting.has(mention))) return;
 | 
			
		||||
		if (note.mentions?.some(mention => this.userIdsWhoMeMuting.has(mention))) return;
 | 
			
		||||
 | 
			
		||||
		if (note.renote && !note.text && isUserRelated(note, this.userIdsWhoMeMutingRenotes)) return;
 | 
			
		||||
 | 
			
		||||
		if (this.user && note.renoteId && !note.text) {
 | 
			
		||||
		if (this.user && isRenotePacked(note) && !isQuotePacked(note)) {
 | 
			
		||||
			if (note.renote && Object.keys(note.renote.reactions).length > 0) {
 | 
			
		||||
				const myRenoteReaction = await this.noteEntityService.populateMyReaction(note.renote, this.user.id);
 | 
			
		||||
				note.renote.myReaction = myRenoteReaction;
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -4,8 +4,6 @@
 | 
			
		|||
 */
 | 
			
		||||
 | 
			
		||||
import { Injectable } from '@nestjs/common';
 | 
			
		||||
import { isUserRelated } from '@/misc/is-user-related.js';
 | 
			
		||||
import type { Packed } from '@/misc/json-schema.js';
 | 
			
		||||
import { NoteEntityService } from '@/core/entities/NoteEntityService.js';
 | 
			
		||||
import { bindThis } from '@/decorators.js';
 | 
			
		||||
import { RoleService } from '@/core/RoleService.js';
 | 
			
		||||
| 
						 | 
				
			
			@ -46,12 +44,7 @@ class RoleTimelineChannel extends Channel {
 | 
			
		|||
			}
 | 
			
		||||
			if (note.visibility !== 'public') return;
 | 
			
		||||
 | 
			
		||||
			// 流れてきたNoteがミュートしているユーザーが関わるものだったら無視する
 | 
			
		||||
			if (isUserRelated(note, this.userIdsWhoMeMuting)) return;
 | 
			
		||||
			// 流れてきたNoteがブロックされているユーザーが関わるものだったら無視する
 | 
			
		||||
			if (isUserRelated(note, this.userIdsWhoBlockingMe)) return;
 | 
			
		||||
 | 
			
		||||
			if (note.renote && !note.text && isUserRelated(note, this.userIdsWhoMeMutingRenotes)) return;
 | 
			
		||||
			if (this.isNoteMutedOrBlocked(note)) return;
 | 
			
		||||
 | 
			
		||||
			this.send('note', note);
 | 
			
		||||
		} else {
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -5,12 +5,11 @@
 | 
			
		|||
 | 
			
		||||
import { Inject, Injectable } from '@nestjs/common';
 | 
			
		||||
import type { MiUserListMembership, UserListMembershipsRepository, UserListsRepository } from '@/models/_.js';
 | 
			
		||||
import { isUserRelated } from '@/misc/is-user-related.js';
 | 
			
		||||
import type { Packed } from '@/misc/json-schema.js';
 | 
			
		||||
import { NoteEntityService } from '@/core/entities/NoteEntityService.js';
 | 
			
		||||
import { DI } from '@/di-symbols.js';
 | 
			
		||||
import { bindThis } from '@/decorators.js';
 | 
			
		||||
import { isInstanceMuted } from '@/misc/is-instance-muted.js';
 | 
			
		||||
import { isRenotePacked, isQuotePacked } from '@/misc/is-renote.js';
 | 
			
		||||
import Channel, { type MiChannelService } from '../channel.js';
 | 
			
		||||
 | 
			
		||||
class UserListChannel extends Channel {
 | 
			
		||||
| 
						 | 
				
			
			@ -106,25 +105,17 @@ class UserListChannel extends Channel {
 | 
			
		|||
			}
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		if (note.renote && note.text == null && (note.fileIds == null || note.fileIds.length === 0) && !this.withRenotes) return;
 | 
			
		||||
		if (isRenotePacked(note) && !isQuotePacked(note) && !this.withRenotes) return;
 | 
			
		||||
 | 
			
		||||
		// 流れてきたNoteがミュートしているユーザーが関わるものだったら無視する
 | 
			
		||||
		if (isUserRelated(note, this.userIdsWhoMeMuting)) return;
 | 
			
		||||
		// 流れてきたNoteがブロックされているユーザーが関わるものだったら無視する
 | 
			
		||||
		if (isUserRelated(note, this.userIdsWhoBlockingMe)) return;
 | 
			
		||||
		if (this.isNoteMutedOrBlocked(note)) return;
 | 
			
		||||
 | 
			
		||||
		if (note.renote && !note.text && isUserRelated(note, this.userIdsWhoMeMutingRenotes)) return;
 | 
			
		||||
 | 
			
		||||
		if (this.user && note.renoteId && !note.text) {
 | 
			
		||||
		if (this.user && isRenotePacked(note) && !isQuotePacked(note)) {
 | 
			
		||||
			if (note.renote && Object.keys(note.renote.reactions).length > 0) {
 | 
			
		||||
				const myRenoteReaction = await this.noteEntityService.populateMyReaction(note.renote, this.user.id);
 | 
			
		||||
				note.renote.myReaction = myRenoteReaction;
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		// 流れてきたNoteがミュートしているインスタンスに関わるものだったら無視する
 | 
			
		||||
		if (isInstanceMuted(note, this.userMutedInstances)) return;
 | 
			
		||||
 | 
			
		||||
		this.connection.cacheNote(note);
 | 
			
		||||
 | 
			
		||||
		this.send('note', note);
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -63,6 +63,22 @@ describe('Renote Mute', () => {
 | 
			
		|||
		assert.strictEqual(res.body.some((note: any) => note.id === carolNote.id), true);
 | 
			
		||||
	});
 | 
			
		||||
 | 
			
		||||
	// #12956
 | 
			
		||||
	test('タイムラインにリノートミュートしているユーザーの通常ノートのリノートが含まれる', async () => {
 | 
			
		||||
		const carolNote = await post(carol, { text: 'hi' });
 | 
			
		||||
		const bobRenote = await post(bob, { renoteId: carolNote.id });
 | 
			
		||||
 | 
			
		||||
		// redisに追加されるのを待つ
 | 
			
		||||
		await sleep(100);
 | 
			
		||||
 | 
			
		||||
		const res = await api('notes/local-timeline', {}, alice);
 | 
			
		||||
 | 
			
		||||
		assert.strictEqual(res.status, 200);
 | 
			
		||||
		assert.strictEqual(Array.isArray(res.body), true);
 | 
			
		||||
		assert.strictEqual(res.body.some((note: any) => note.id === carolNote.id), true);
 | 
			
		||||
		assert.strictEqual(res.body.some((note: any) => note.id === bobRenote.id), true);
 | 
			
		||||
	});
 | 
			
		||||
 | 
			
		||||
	test('ストリームにリノートミュートしているユーザーのリノートが流れない', async () => {
 | 
			
		||||
		const bobNote = await post(bob, { text: 'hi' });
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -86,4 +102,17 @@ describe('Renote Mute', () => {
 | 
			
		|||
 | 
			
		||||
		assert.strictEqual(fired, true);
 | 
			
		||||
	});
 | 
			
		||||
 | 
			
		||||
	// #12956
 | 
			
		||||
	test('ストリームにリノートミュートしているユーザーの通常ノートのリノートが流れてくる', async () => {
 | 
			
		||||
		const carolbNote = await post(carol, { text: 'hi' });
 | 
			
		||||
 | 
			
		||||
		const fired = await waitFire(
 | 
			
		||||
			alice, 'localTimeline',
 | 
			
		||||
			() => api('notes/create', { renoteId: carolbNote.id }, bob),
 | 
			
		||||
			msg => msg.type === 'note' && msg.body.userId === bob.id,
 | 
			
		||||
		);
 | 
			
		||||
 | 
			
		||||
		assert.strictEqual(fired, true);
 | 
			
		||||
	});
 | 
			
		||||
});
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -63,7 +63,7 @@ describe('Streaming', () => {
 | 
			
		|||
			takumiNote = await post(takumi, { text: 'piyo' });
 | 
			
		||||
 | 
			
		||||
			// Follow: ayano => kyoko
 | 
			
		||||
			await api('following/create', { userId: kyoko.id }, ayano);
 | 
			
		||||
			await api('following/create', { userId: kyoko.id, withReplies: false }, ayano);
 | 
			
		||||
 | 
			
		||||
			// Follow: ayano => akari
 | 
			
		||||
			await follow(ayano, akari);
 | 
			
		||||
| 
						 | 
				
			
			@ -509,6 +509,16 @@ describe('Streaming', () => {
 | 
			
		|||
 | 
			
		||||
				assert.strictEqual(fired, false);
 | 
			
		||||
			});
 | 
			
		||||
 | 
			
		||||
			test('withReplies = falseでフォローしてる人によるリプライが流れてくる', async () => {
 | 
			
		||||
				const fired = await waitFire(
 | 
			
		||||
					ayano, 'globalTimeline',		// ayano:Global
 | 
			
		||||
					() => api('notes/create', { text: 'foo', replyId: kanakoNote.id }, kyoko),	// kyoko posts
 | 
			
		||||
					msg => msg.type === 'note' && msg.body.userId === kyoko.id,	// wait kyoko
 | 
			
		||||
				);
 | 
			
		||||
 | 
			
		||||
				assert.strictEqual(fired, true);
 | 
			
		||||
			});
 | 
			
		||||
		});
 | 
			
		||||
 | 
			
		||||
		describe('UserList Timeline', () => {
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										144
									
								
								packages/backend/test/unit/NoteCreateService.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										144
									
								
								packages/backend/test/unit/NoteCreateService.ts
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,144 @@
 | 
			
		|||
/*
 | 
			
		||||
 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
			
		||||
 * SPDX-License-Identifier: AGPL-3.0-only
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
import { Test } from '@nestjs/testing';
 | 
			
		||||
 | 
			
		||||
import { CoreModule } from '@/core/CoreModule.js';
 | 
			
		||||
import { NoteCreateService } from '@/core/NoteCreateService.js';
 | 
			
		||||
import { GlobalModule } from '@/GlobalModule.js';
 | 
			
		||||
import { MiNote } from '@/models/Note.js';
 | 
			
		||||
import { IPoll } from '@/models/Poll.js';
 | 
			
		||||
import { MiDriveFile } from '@/models/DriveFile.js';
 | 
			
		||||
 | 
			
		||||
describe('NoteCreateService', () => {
 | 
			
		||||
	let noteCreateService: NoteCreateService;
 | 
			
		||||
 | 
			
		||||
	beforeAll(async () => {
 | 
			
		||||
		const app = await Test.createTestingModule({
 | 
			
		||||
			imports: [GlobalModule, CoreModule],
 | 
			
		||||
		}).compile();
 | 
			
		||||
		noteCreateService = app.get<NoteCreateService>(NoteCreateService);
 | 
			
		||||
	});
 | 
			
		||||
 | 
			
		||||
	describe('is-renote', () => {
 | 
			
		||||
		const base: MiNote = {
 | 
			
		||||
			id: 'some-note-id',
 | 
			
		||||
			replyId: null,
 | 
			
		||||
			reply: null,
 | 
			
		||||
			renoteId: null,
 | 
			
		||||
			renote: null,
 | 
			
		||||
			threadId: null,
 | 
			
		||||
			text: null,
 | 
			
		||||
			name: null,
 | 
			
		||||
			cw: null,
 | 
			
		||||
			userId: 'some-user-id',
 | 
			
		||||
			user: null,
 | 
			
		||||
			localOnly: false,
 | 
			
		||||
			reactionAcceptance: null,
 | 
			
		||||
			renoteCount: 0,
 | 
			
		||||
			repliesCount: 0,
 | 
			
		||||
			clippedCount: 0,
 | 
			
		||||
			reactions: {},
 | 
			
		||||
			visibility: 'public',
 | 
			
		||||
			uri: null,
 | 
			
		||||
			url: null,
 | 
			
		||||
			fileIds: [],
 | 
			
		||||
			attachedFileTypes: [],
 | 
			
		||||
			visibleUserIds: [],
 | 
			
		||||
			mentions: [],
 | 
			
		||||
			mentionedRemoteUsers: '',
 | 
			
		||||
			reactionAndUserPairCache: [],
 | 
			
		||||
			emojis: [],
 | 
			
		||||
			tags: [],
 | 
			
		||||
			hasPoll: false,
 | 
			
		||||
			channelId: null,
 | 
			
		||||
			channel: null,
 | 
			
		||||
			userHost: null,
 | 
			
		||||
			replyUserId: null,
 | 
			
		||||
			replyUserHost: null,
 | 
			
		||||
			renoteUserId: null,
 | 
			
		||||
			renoteUserHost: null,
 | 
			
		||||
		};
 | 
			
		||||
 | 
			
		||||
		const poll: IPoll = {
 | 
			
		||||
			choices: ['kinoko', 'takenoko'],
 | 
			
		||||
			multiple: false,
 | 
			
		||||
			expiresAt: null,
 | 
			
		||||
		};
 | 
			
		||||
 | 
			
		||||
		const file: MiDriveFile = {
 | 
			
		||||
			id: 'some-file-id',
 | 
			
		||||
			userId: null,
 | 
			
		||||
			user: null,
 | 
			
		||||
			userHost: null,
 | 
			
		||||
			md5: '',
 | 
			
		||||
			name: '',
 | 
			
		||||
			type: '',
 | 
			
		||||
			size: 0,
 | 
			
		||||
			comment: null,
 | 
			
		||||
			blurhash: null,
 | 
			
		||||
			properties: {},
 | 
			
		||||
			storedInternal: false,
 | 
			
		||||
			url: '',
 | 
			
		||||
			thumbnailUrl: null,
 | 
			
		||||
			webpublicUrl: null,
 | 
			
		||||
			webpublicType: null,
 | 
			
		||||
			accessKey: null,
 | 
			
		||||
			thumbnailAccessKey: null,
 | 
			
		||||
			webpublicAccessKey: null,
 | 
			
		||||
			uri: null,
 | 
			
		||||
			src: null,
 | 
			
		||||
			folderId: null,
 | 
			
		||||
			folder: null,
 | 
			
		||||
			isSensitive: false,
 | 
			
		||||
			maybeSensitive: false,
 | 
			
		||||
			maybePorn: false,
 | 
			
		||||
			isLink: false,
 | 
			
		||||
			requestHeaders: null,
 | 
			
		||||
			requestIp: null,
 | 
			
		||||
		};
 | 
			
		||||
 | 
			
		||||
		test('note without renote should not be Renote', () => {
 | 
			
		||||
			const note = { renote: null };
 | 
			
		||||
			expect(noteCreateService['isRenote'](note)).toBe(false);
 | 
			
		||||
		});
 | 
			
		||||
 | 
			
		||||
		test('note with renote should be Renote and not be Quote', () => {
 | 
			
		||||
			const note = { renote: base };
 | 
			
		||||
			expect(noteCreateService['isRenote'](note)).toBe(true);
 | 
			
		||||
			expect(noteCreateService['isQuote'](note)).toBe(false);
 | 
			
		||||
		});
 | 
			
		||||
 | 
			
		||||
		test('note with renote and text should be Quote', () => {
 | 
			
		||||
			const note = { renote: base, text: 'some-text' };
 | 
			
		||||
			expect(noteCreateService['isRenote'](note)).toBe(true);
 | 
			
		||||
			expect(noteCreateService['isQuote'](note)).toBe(true);
 | 
			
		||||
		});
 | 
			
		||||
 | 
			
		||||
		test('note with renote and cw should be Quote', () => {
 | 
			
		||||
			const note = { renote: base, cw: 'some-cw' };
 | 
			
		||||
			expect(noteCreateService['isRenote'](note)).toBe(true);
 | 
			
		||||
			expect(noteCreateService['isQuote'](note)).toBe(true);
 | 
			
		||||
		});
 | 
			
		||||
 | 
			
		||||
		test('note with renote and reply should be Quote', () => {
 | 
			
		||||
			const note = { renote: base, reply: { ...base, id: 'another-note-id' } };
 | 
			
		||||
			expect(noteCreateService['isRenote'](note)).toBe(true);
 | 
			
		||||
			expect(noteCreateService['isQuote'](note)).toBe(true);
 | 
			
		||||
		});
 | 
			
		||||
 | 
			
		||||
		test('note with renote and poll should be Quote', () => {
 | 
			
		||||
			const note = { renote: base, poll };
 | 
			
		||||
			expect(noteCreateService['isRenote'](note)).toBe(true);
 | 
			
		||||
			expect(noteCreateService['isQuote'](note)).toBe(true);
 | 
			
		||||
		});
 | 
			
		||||
 | 
			
		||||
		test('note with renote and non-empty files should be Quote', () => {
 | 
			
		||||
			const note = { renote: base, files: [file] };
 | 
			
		||||
			expect(noteCreateService['isRenote'](note)).toBe(true);
 | 
			
		||||
			expect(noteCreateService['isQuote'](note)).toBe(true);
 | 
			
		||||
		});
 | 
			
		||||
	});
 | 
			
		||||
});
 | 
			
		||||
| 
						 | 
				
			
			@ -3,6 +3,8 @@
 | 
			
		|||
 * SPDX-License-Identifier: AGPL-3.0-only
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
import { UserEntityService } from '@/core/entities/UserEntityService.js';
 | 
			
		||||
 | 
			
		||||
process.env.NODE_ENV = 'test';
 | 
			
		||||
 | 
			
		||||
import { jest } from '@jest/globals';
 | 
			
		||||
| 
						 | 
				
			
			@ -20,6 +22,7 @@ import { IdService } from '@/core/IdService.js';
 | 
			
		|||
import { GlobalEventService } from '@/core/GlobalEventService.js';
 | 
			
		||||
import { secureRndstr } from '@/misc/secure-rndstr.js';
 | 
			
		||||
import { NotificationService } from '@/core/NotificationService.js';
 | 
			
		||||
import { RoleCondFormulaValue } from '@/models/Role.js';
 | 
			
		||||
import { sleep } from '../utils.js';
 | 
			
		||||
import type { TestingModule } from '@nestjs/testing';
 | 
			
		||||
import type { MockFunctionMetadata } from 'jest-mock';
 | 
			
		||||
| 
						 | 
				
			
			@ -52,12 +55,26 @@ describe('RoleService', () => {
 | 
			
		|||
			id: genAidx(Date.now()),
 | 
			
		||||
			updatedAt: new Date(),
 | 
			
		||||
			lastUsedAt: new Date(),
 | 
			
		||||
			name: '',
 | 
			
		||||
			description: '',
 | 
			
		||||
			...data,
 | 
			
		||||
		})
 | 
			
		||||
			.then(x => rolesRepository.findOneByOrFail(x.identifiers[0]));
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	function createConditionalRole(condFormula: RoleCondFormulaValue, data: Partial<MiRole> = {}) {
 | 
			
		||||
		return createRole({
 | 
			
		||||
			name: `[conditional] ${condFormula.type}`,
 | 
			
		||||
			target: 'conditional',
 | 
			
		||||
			condFormula: condFormula,
 | 
			
		||||
			...data,
 | 
			
		||||
		});
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	function aidx() {
 | 
			
		||||
		return genAidx(Date.now());
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	beforeEach(async () => {
 | 
			
		||||
		clock = lolex.install({
 | 
			
		||||
			now: new Date(),
 | 
			
		||||
| 
						 | 
				
			
			@ -73,6 +90,7 @@ describe('RoleService', () => {
 | 
			
		|||
				CacheService,
 | 
			
		||||
				IdService,
 | 
			
		||||
				GlobalEventService,
 | 
			
		||||
				UserEntityService,
 | 
			
		||||
				{
 | 
			
		||||
					provide: NotificationService,
 | 
			
		||||
					useFactory: () => ({
 | 
			
		||||
| 
						 | 
				
			
			@ -209,79 +227,6 @@ describe('RoleService', () => {
 | 
			
		|||
			expect(result.driveCapacityMb).toBe(100);
 | 
			
		||||
		});
 | 
			
		||||
 | 
			
		||||
		test('conditional role', async () => {
 | 
			
		||||
			const user1 = await createUser({
 | 
			
		||||
				id: genAidx(Date.now() - (1000 * 60 * 60 * 24 * 365)),
 | 
			
		||||
			});
 | 
			
		||||
			const user2 = await createUser({
 | 
			
		||||
				id: genAidx(Date.now() - (1000 * 60 * 60 * 24 * 365)),
 | 
			
		||||
				followersCount: 10,
 | 
			
		||||
			});
 | 
			
		||||
			await createRole({
 | 
			
		||||
				name: 'a',
 | 
			
		||||
				policies: {
 | 
			
		||||
					canManageCustomEmojis: {
 | 
			
		||||
						useDefault: false,
 | 
			
		||||
						priority: 0,
 | 
			
		||||
						value: true,
 | 
			
		||||
					},
 | 
			
		||||
				},
 | 
			
		||||
				target: 'conditional',
 | 
			
		||||
				condFormula: {
 | 
			
		||||
					id: '232a4221-9816-49a6-a967-ae0fac52ec5e',
 | 
			
		||||
					type: 'and',
 | 
			
		||||
					values: [{
 | 
			
		||||
						id: '2a37ef43-2d93-4c4d-87f6-f2fdb7d9b530',
 | 
			
		||||
						type: 'followersMoreThanOrEq',
 | 
			
		||||
						value: 10,
 | 
			
		||||
					}, {
 | 
			
		||||
						id: '1bd67839-b126-4f92-bad0-4e285dab453b',
 | 
			
		||||
						type: 'createdMoreThan',
 | 
			
		||||
						sec: 60 * 60 * 24 * 7,
 | 
			
		||||
					}],
 | 
			
		||||
				},
 | 
			
		||||
			});
 | 
			
		||||
 | 
			
		||||
			metaService.fetch.mockResolvedValue({
 | 
			
		||||
				policies: {
 | 
			
		||||
					canManageCustomEmojis: false,
 | 
			
		||||
				},
 | 
			
		||||
			} as any);
 | 
			
		||||
 | 
			
		||||
			const user1Policies = await roleService.getUserPolicies(user1.id);
 | 
			
		||||
			const user2Policies = await roleService.getUserPolicies(user2.id);
 | 
			
		||||
			expect(user1Policies.canManageCustomEmojis).toBe(false);
 | 
			
		||||
			expect(user2Policies.canManageCustomEmojis).toBe(true);
 | 
			
		||||
		});
 | 
			
		||||
 | 
			
		||||
		test('コンディショナルロール: マニュアルロールにアサイン済み', async () => {
 | 
			
		||||
			const [user1, user2, role1] = await Promise.all([
 | 
			
		||||
				createUser(),
 | 
			
		||||
				createUser(),
 | 
			
		||||
				createRole({
 | 
			
		||||
					name: 'manual role',
 | 
			
		||||
				}),
 | 
			
		||||
			]);
 | 
			
		||||
			const role2 = await createRole({
 | 
			
		||||
				name: 'conditional role',
 | 
			
		||||
				target: 'conditional',
 | 
			
		||||
				condFormula: {
 | 
			
		||||
					// idはバックエンドのロジックに必要ない?
 | 
			
		||||
					id: 'bdc612bd-9d54-4675-ae83-0499c82ea670',
 | 
			
		||||
					type: 'roleAssignedTo',
 | 
			
		||||
					roleId: role1.id,
 | 
			
		||||
				},
 | 
			
		||||
			});
 | 
			
		||||
			await roleService.assign(user2.id, role1.id);
 | 
			
		||||
 | 
			
		||||
			const [u1role, u2role] = await Promise.all([
 | 
			
		||||
				roleService.getUserRoles(user1.id),
 | 
			
		||||
				roleService.getUserRoles(user2.id),
 | 
			
		||||
			]);
 | 
			
		||||
			expect(u1role.some(r => r.id === role2.id)).toBe(false);
 | 
			
		||||
			expect(u2role.some(r => r.id === role2.id)).toBe(true);
 | 
			
		||||
		});
 | 
			
		||||
 | 
			
		||||
		test('expired role', async () => {
 | 
			
		||||
			const user = await createUser();
 | 
			
		||||
			const role = await createRole({
 | 
			
		||||
| 
						 | 
				
			
			@ -320,6 +265,427 @@ describe('RoleService', () => {
 | 
			
		|||
		});
 | 
			
		||||
	});
 | 
			
		||||
 | 
			
		||||
	describe('conditional role', () => {
 | 
			
		||||
		test('~かつ~', async () => {
 | 
			
		||||
			const [user1, user2, user3, user4] = await Promise.all([
 | 
			
		||||
				createUser({ isBot: true, isCat: false, isSuspended: false }),
 | 
			
		||||
				createUser({ isBot: false, isCat: true, isSuspended: false }),
 | 
			
		||||
				createUser({ isBot: true, isCat: true, isSuspended: false }),
 | 
			
		||||
				createUser({ isBot: false, isCat: false, isSuspended: true }),
 | 
			
		||||
			]);
 | 
			
		||||
			const role1 = await createConditionalRole({
 | 
			
		||||
				id: aidx(),
 | 
			
		||||
				type: 'isBot',
 | 
			
		||||
			});
 | 
			
		||||
			const role2 = await createConditionalRole({
 | 
			
		||||
				id: aidx(),
 | 
			
		||||
				type: 'isCat',
 | 
			
		||||
			});
 | 
			
		||||
			const role3 = await createConditionalRole({
 | 
			
		||||
				id: aidx(),
 | 
			
		||||
				type: 'isSuspended',
 | 
			
		||||
			});
 | 
			
		||||
			const role4 = await createConditionalRole({
 | 
			
		||||
				id: aidx(),
 | 
			
		||||
				type: 'and',
 | 
			
		||||
				values: [role1.condFormula, role2.condFormula],
 | 
			
		||||
			});
 | 
			
		||||
 | 
			
		||||
			const actual1 = await roleService.getUserRoles(user1.id);
 | 
			
		||||
			const actual2 = await roleService.getUserRoles(user2.id);
 | 
			
		||||
			const actual3 = await roleService.getUserRoles(user3.id);
 | 
			
		||||
			const actual4 = await roleService.getUserRoles(user4.id);
 | 
			
		||||
			expect(actual1.some(r => r.id === role4.id)).toBe(false);
 | 
			
		||||
			expect(actual2.some(r => r.id === role4.id)).toBe(false);
 | 
			
		||||
			expect(actual3.some(r => r.id === role4.id)).toBe(true);
 | 
			
		||||
			expect(actual4.some(r => r.id === role4.id)).toBe(false);
 | 
			
		||||
		});
 | 
			
		||||
 | 
			
		||||
		test('~または~', async () => {
 | 
			
		||||
			const [user1, user2, user3, user4] = await Promise.all([
 | 
			
		||||
				createUser({ isBot: true, isCat: false, isSuspended: false }),
 | 
			
		||||
				createUser({ isBot: false, isCat: true, isSuspended: false }),
 | 
			
		||||
				createUser({ isBot: true, isCat: true, isSuspended: false }),
 | 
			
		||||
				createUser({ isBot: false, isCat: false, isSuspended: true }),
 | 
			
		||||
			]);
 | 
			
		||||
			const role1 = await createConditionalRole({
 | 
			
		||||
				id: aidx(),
 | 
			
		||||
				type: 'isBot',
 | 
			
		||||
			});
 | 
			
		||||
			const role2 = await createConditionalRole({
 | 
			
		||||
				id: aidx(),
 | 
			
		||||
				type: 'isCat',
 | 
			
		||||
			});
 | 
			
		||||
			const role3 = await createConditionalRole({
 | 
			
		||||
				id: aidx(),
 | 
			
		||||
				type: 'isSuspended',
 | 
			
		||||
			});
 | 
			
		||||
			const role4 = await createConditionalRole({
 | 
			
		||||
				id: aidx(),
 | 
			
		||||
				type: 'or',
 | 
			
		||||
				values: [role1.condFormula, role2.condFormula],
 | 
			
		||||
			});
 | 
			
		||||
 | 
			
		||||
			const actual1 = await roleService.getUserRoles(user1.id);
 | 
			
		||||
			const actual2 = await roleService.getUserRoles(user2.id);
 | 
			
		||||
			const actual3 = await roleService.getUserRoles(user3.id);
 | 
			
		||||
			const actual4 = await roleService.getUserRoles(user4.id);
 | 
			
		||||
			expect(actual1.some(r => r.id === role4.id)).toBe(true);
 | 
			
		||||
			expect(actual2.some(r => r.id === role4.id)).toBe(true);
 | 
			
		||||
			expect(actual3.some(r => r.id === role4.id)).toBe(true);
 | 
			
		||||
			expect(actual4.some(r => r.id === role4.id)).toBe(false);
 | 
			
		||||
		});
 | 
			
		||||
 | 
			
		||||
		test('~ではない', async () => {
 | 
			
		||||
			const [user1, user2, user3] = await Promise.all([
 | 
			
		||||
				createUser({ isBot: true, isCat: false, isSuspended: false }),
 | 
			
		||||
				createUser({ isBot: false, isCat: true, isSuspended: false }),
 | 
			
		||||
				createUser({ isBot: true, isCat: true, isSuspended: false }),
 | 
			
		||||
			]);
 | 
			
		||||
			const role1 = await createConditionalRole({
 | 
			
		||||
				id: aidx(),
 | 
			
		||||
				type: 'isBot',
 | 
			
		||||
			});
 | 
			
		||||
			const role2 = await createConditionalRole({
 | 
			
		||||
				id: aidx(),
 | 
			
		||||
				type: 'isCat',
 | 
			
		||||
			});
 | 
			
		||||
			const role4 = await createConditionalRole({
 | 
			
		||||
				id: aidx(),
 | 
			
		||||
				type: 'not',
 | 
			
		||||
				value: role1.condFormula,
 | 
			
		||||
			});
 | 
			
		||||
 | 
			
		||||
			const actual1 = await roleService.getUserRoles(user1.id);
 | 
			
		||||
			const actual2 = await roleService.getUserRoles(user2.id);
 | 
			
		||||
			const actual3 = await roleService.getUserRoles(user3.id);
 | 
			
		||||
			expect(actual1.some(r => r.id === role4.id)).toBe(false);
 | 
			
		||||
			expect(actual2.some(r => r.id === role4.id)).toBe(true);
 | 
			
		||||
			expect(actual3.some(r => r.id === role4.id)).toBe(false);
 | 
			
		||||
		});
 | 
			
		||||
 | 
			
		||||
		test('マニュアルロールにアサイン済み', async () => {
 | 
			
		||||
			const [user1, user2, role1] = await Promise.all([
 | 
			
		||||
				createUser(),
 | 
			
		||||
				createUser(),
 | 
			
		||||
				createRole({
 | 
			
		||||
					name: 'manual role',
 | 
			
		||||
				}),
 | 
			
		||||
			]);
 | 
			
		||||
			const role2 = await createConditionalRole({
 | 
			
		||||
				id: aidx(),
 | 
			
		||||
				type: 'roleAssignedTo',
 | 
			
		||||
				roleId: role1.id,
 | 
			
		||||
			});
 | 
			
		||||
			await roleService.assign(user2.id, role1.id);
 | 
			
		||||
 | 
			
		||||
			const [u1role, u2role] = await Promise.all([
 | 
			
		||||
				roleService.getUserRoles(user1.id),
 | 
			
		||||
				roleService.getUserRoles(user2.id),
 | 
			
		||||
			]);
 | 
			
		||||
			expect(u1role.some(r => r.id === role2.id)).toBe(false);
 | 
			
		||||
			expect(u2role.some(r => r.id === role2.id)).toBe(true);
 | 
			
		||||
		});
 | 
			
		||||
 | 
			
		||||
		test('ローカルユーザのみ', async () => {
 | 
			
		||||
			const [user1, user2] = await Promise.all([
 | 
			
		||||
				createUser({ host: null }),
 | 
			
		||||
				createUser({ host: 'example.com' }),
 | 
			
		||||
			]);
 | 
			
		||||
			const role = await createConditionalRole({
 | 
			
		||||
				id: aidx(),
 | 
			
		||||
				type: 'isLocal',
 | 
			
		||||
			});
 | 
			
		||||
 | 
			
		||||
			const actual1 = await roleService.getUserRoles(user1.id);
 | 
			
		||||
			const actual2 = await roleService.getUserRoles(user2.id);
 | 
			
		||||
			expect(actual1.some(r => r.id === role.id)).toBe(true);
 | 
			
		||||
			expect(actual2.some(r => r.id === role.id)).toBe(false);
 | 
			
		||||
		});
 | 
			
		||||
 | 
			
		||||
		test('リモートユーザのみ', async () => {
 | 
			
		||||
			const [user1, user2] = await Promise.all([
 | 
			
		||||
				createUser({ host: null }),
 | 
			
		||||
				createUser({ host: 'example.com' }),
 | 
			
		||||
			]);
 | 
			
		||||
			const role = await createConditionalRole({
 | 
			
		||||
				id: aidx(),
 | 
			
		||||
				type: 'isRemote',
 | 
			
		||||
			});
 | 
			
		||||
 | 
			
		||||
			const actual1 = await roleService.getUserRoles(user1.id);
 | 
			
		||||
			const actual2 = await roleService.getUserRoles(user2.id);
 | 
			
		||||
			expect(actual1.some(r => r.id === role.id)).toBe(false);
 | 
			
		||||
			expect(actual2.some(r => r.id === role.id)).toBe(true);
 | 
			
		||||
		});
 | 
			
		||||
 | 
			
		||||
		test('サスペンド済みユーザである', async () => {
 | 
			
		||||
			const [user1, user2] = await Promise.all([
 | 
			
		||||
				createUser({ isSuspended: false }),
 | 
			
		||||
				createUser({ isSuspended: true }),
 | 
			
		||||
			]);
 | 
			
		||||
			const role = await createConditionalRole({
 | 
			
		||||
				id: aidx(),
 | 
			
		||||
				type: 'isSuspended',
 | 
			
		||||
			});
 | 
			
		||||
 | 
			
		||||
			const actual1 = await roleService.getUserRoles(user1.id);
 | 
			
		||||
			const actual2 = await roleService.getUserRoles(user2.id);
 | 
			
		||||
			expect(actual1.some(r => r.id === role.id)).toBe(false);
 | 
			
		||||
			expect(actual2.some(r => r.id === role.id)).toBe(true);
 | 
			
		||||
		});
 | 
			
		||||
 | 
			
		||||
		test('鍵アカウントユーザである', async () => {
 | 
			
		||||
			const [user1, user2] = await Promise.all([
 | 
			
		||||
				createUser({ isLocked: false }),
 | 
			
		||||
				createUser({ isLocked: true }),
 | 
			
		||||
			]);
 | 
			
		||||
			const role = await createConditionalRole({
 | 
			
		||||
				id: aidx(),
 | 
			
		||||
				type: 'isLocked',
 | 
			
		||||
			});
 | 
			
		||||
 | 
			
		||||
			const actual1 = await roleService.getUserRoles(user1.id);
 | 
			
		||||
			const actual2 = await roleService.getUserRoles(user2.id);
 | 
			
		||||
			expect(actual1.some(r => r.id === role.id)).toBe(false);
 | 
			
		||||
			expect(actual2.some(r => r.id === role.id)).toBe(true);
 | 
			
		||||
		});
 | 
			
		||||
 | 
			
		||||
		test('botユーザである', async () => {
 | 
			
		||||
			const [user1, user2] = await Promise.all([
 | 
			
		||||
				createUser({ isBot: false }),
 | 
			
		||||
				createUser({ isBot: true }),
 | 
			
		||||
			]);
 | 
			
		||||
			const role = await createConditionalRole({
 | 
			
		||||
				id: aidx(),
 | 
			
		||||
				type: 'isBot',
 | 
			
		||||
			});
 | 
			
		||||
 | 
			
		||||
			const actual1 = await roleService.getUserRoles(user1.id);
 | 
			
		||||
			const actual2 = await roleService.getUserRoles(user2.id);
 | 
			
		||||
			expect(actual1.some(r => r.id === role.id)).toBe(false);
 | 
			
		||||
			expect(actual2.some(r => r.id === role.id)).toBe(true);
 | 
			
		||||
		});
 | 
			
		||||
 | 
			
		||||
		test('猫である', async () => {
 | 
			
		||||
			const [user1, user2] = await Promise.all([
 | 
			
		||||
				createUser({ isCat: false }),
 | 
			
		||||
				createUser({ isCat: true }),
 | 
			
		||||
			]);
 | 
			
		||||
			const role = await createConditionalRole({
 | 
			
		||||
				id: aidx(),
 | 
			
		||||
				type: 'isCat',
 | 
			
		||||
			});
 | 
			
		||||
 | 
			
		||||
			const actual1 = await roleService.getUserRoles(user1.id);
 | 
			
		||||
			const actual2 = await roleService.getUserRoles(user2.id);
 | 
			
		||||
			expect(actual1.some(r => r.id === role.id)).toBe(false);
 | 
			
		||||
			expect(actual2.some(r => r.id === role.id)).toBe(true);
 | 
			
		||||
		});
 | 
			
		||||
 | 
			
		||||
		test('「ユーザを見つけやすくする」が有効なアカウント', async () => {
 | 
			
		||||
			const [user1, user2] = await Promise.all([
 | 
			
		||||
				createUser({ isExplorable: false }),
 | 
			
		||||
				createUser({ isExplorable: true }),
 | 
			
		||||
			]);
 | 
			
		||||
			const role = await createConditionalRole({
 | 
			
		||||
				id: aidx(),
 | 
			
		||||
				type: 'isExplorable',
 | 
			
		||||
			});
 | 
			
		||||
 | 
			
		||||
			const actual1 = await roleService.getUserRoles(user1.id);
 | 
			
		||||
			const actual2 = await roleService.getUserRoles(user2.id);
 | 
			
		||||
			expect(actual1.some(r => r.id === role.id)).toBe(false);
 | 
			
		||||
			expect(actual2.some(r => r.id === role.id)).toBe(true);
 | 
			
		||||
		});
 | 
			
		||||
 | 
			
		||||
		test('ユーザが作成されてから指定期間経過した', async () => {
 | 
			
		||||
			const base = new Date();
 | 
			
		||||
			base.setMinutes(base.getMinutes() - 5);
 | 
			
		||||
 | 
			
		||||
			const d1 = new Date(base);
 | 
			
		||||
			const d2 = new Date(base);
 | 
			
		||||
			const d3 = new Date(base);
 | 
			
		||||
			d1.setSeconds(d1.getSeconds() - 1);
 | 
			
		||||
			d3.setSeconds(d3.getSeconds() + 1);
 | 
			
		||||
 | 
			
		||||
			const [user1, user2, user3] = await Promise.all([
 | 
			
		||||
				// 4:59
 | 
			
		||||
				createUser({ id: genAidx(d1.getTime()) }),
 | 
			
		||||
				// 5:00
 | 
			
		||||
				createUser({ id: genAidx(d2.getTime()) }),
 | 
			
		||||
				// 5:01
 | 
			
		||||
				createUser({ id: genAidx(d3.getTime()) }),
 | 
			
		||||
			]);
 | 
			
		||||
			const role = await createConditionalRole({
 | 
			
		||||
				id: aidx(),
 | 
			
		||||
				type: 'createdLessThan',
 | 
			
		||||
				// 5 minutes
 | 
			
		||||
				sec: 300,
 | 
			
		||||
			});
 | 
			
		||||
 | 
			
		||||
			const actual1 = await roleService.getUserRoles(user1.id);
 | 
			
		||||
			const actual2 = await roleService.getUserRoles(user2.id);
 | 
			
		||||
			const actual3 = await roleService.getUserRoles(user3.id);
 | 
			
		||||
			expect(actual1.some(r => r.id === role.id)).toBe(false);
 | 
			
		||||
			expect(actual2.some(r => r.id === role.id)).toBe(false);
 | 
			
		||||
			expect(actual3.some(r => r.id === role.id)).toBe(true);
 | 
			
		||||
		});
 | 
			
		||||
 | 
			
		||||
		test('ユーザが作成されてから指定期間経っていない', async () => {
 | 
			
		||||
			const base = new Date();
 | 
			
		||||
			base.setMinutes(base.getMinutes() - 5);
 | 
			
		||||
 | 
			
		||||
			const d1 = new Date(base);
 | 
			
		||||
			const d2 = new Date(base);
 | 
			
		||||
			const d3 = new Date(base);
 | 
			
		||||
			d1.setSeconds(d1.getSeconds() - 1);
 | 
			
		||||
			d3.setSeconds(d3.getSeconds() + 1);
 | 
			
		||||
 | 
			
		||||
			const [user1, user2, user3] = await Promise.all([
 | 
			
		||||
				// 4:59
 | 
			
		||||
				createUser({ id: genAidx(d1.getTime()) }),
 | 
			
		||||
				// 5:00
 | 
			
		||||
				createUser({ id: genAidx(d2.getTime()) }),
 | 
			
		||||
				// 5:01
 | 
			
		||||
				createUser({ id: genAidx(d3.getTime()) }),
 | 
			
		||||
			]);
 | 
			
		||||
			const role = await createConditionalRole({
 | 
			
		||||
				id: aidx(),
 | 
			
		||||
				type: 'createdMoreThan',
 | 
			
		||||
				// 5 minutes
 | 
			
		||||
				sec: 300,
 | 
			
		||||
			});
 | 
			
		||||
 | 
			
		||||
			const actual1 = await roleService.getUserRoles(user1.id);
 | 
			
		||||
			const actual2 = await roleService.getUserRoles(user2.id);
 | 
			
		||||
			const actual3 = await roleService.getUserRoles(user3.id);
 | 
			
		||||
			expect(actual1.some(r => r.id === role.id)).toBe(true);
 | 
			
		||||
			expect(actual2.some(r => r.id === role.id)).toBe(false);
 | 
			
		||||
			expect(actual3.some(r => r.id === role.id)).toBe(false);
 | 
			
		||||
		});
 | 
			
		||||
 | 
			
		||||
		test('フォロワー数が指定値以下', async () => {
 | 
			
		||||
			const [user1, user2, user3] = await Promise.all([
 | 
			
		||||
				createUser({ followersCount: 99 }),
 | 
			
		||||
				createUser({ followersCount: 100 }),
 | 
			
		||||
				createUser({ followersCount: 101 }),
 | 
			
		||||
			]);
 | 
			
		||||
			const role = await createConditionalRole({
 | 
			
		||||
				id: aidx(),
 | 
			
		||||
				type: 'followersLessThanOrEq',
 | 
			
		||||
				value: 100,
 | 
			
		||||
			});
 | 
			
		||||
 | 
			
		||||
			const actual1 = await roleService.getUserRoles(user1.id);
 | 
			
		||||
			const actual2 = await roleService.getUserRoles(user2.id);
 | 
			
		||||
			const actual3 = await roleService.getUserRoles(user3.id);
 | 
			
		||||
			expect(actual1.some(r => r.id === role.id)).toBe(true);
 | 
			
		||||
			expect(actual2.some(r => r.id === role.id)).toBe(true);
 | 
			
		||||
			expect(actual3.some(r => r.id === role.id)).toBe(false);
 | 
			
		||||
		});
 | 
			
		||||
 | 
			
		||||
		test('フォロワー数が指定値以下', async () => {
 | 
			
		||||
			const [user1, user2, user3] = await Promise.all([
 | 
			
		||||
				createUser({ followersCount: 99 }),
 | 
			
		||||
				createUser({ followersCount: 100 }),
 | 
			
		||||
				createUser({ followersCount: 101 }),
 | 
			
		||||
			]);
 | 
			
		||||
			const role = await createConditionalRole({
 | 
			
		||||
				id: aidx(),
 | 
			
		||||
				type: 'followersMoreThanOrEq',
 | 
			
		||||
				value: 100,
 | 
			
		||||
			});
 | 
			
		||||
 | 
			
		||||
			const actual1 = await roleService.getUserRoles(user1.id);
 | 
			
		||||
			const actual2 = await roleService.getUserRoles(user2.id);
 | 
			
		||||
			const actual3 = await roleService.getUserRoles(user3.id);
 | 
			
		||||
			expect(actual1.some(r => r.id === role.id)).toBe(false);
 | 
			
		||||
			expect(actual2.some(r => r.id === role.id)).toBe(true);
 | 
			
		||||
			expect(actual3.some(r => r.id === role.id)).toBe(true);
 | 
			
		||||
		});
 | 
			
		||||
 | 
			
		||||
		test('フォロー数が指定値以下', async () => {
 | 
			
		||||
			const [user1, user2, user3] = await Promise.all([
 | 
			
		||||
				createUser({ followingCount: 99 }),
 | 
			
		||||
				createUser({ followingCount: 100 }),
 | 
			
		||||
				createUser({ followingCount: 101 }),
 | 
			
		||||
			]);
 | 
			
		||||
			const role = await createConditionalRole({
 | 
			
		||||
				id: aidx(),
 | 
			
		||||
				type: 'followingLessThanOrEq',
 | 
			
		||||
				value: 100,
 | 
			
		||||
			});
 | 
			
		||||
 | 
			
		||||
			const actual1 = await roleService.getUserRoles(user1.id);
 | 
			
		||||
			const actual2 = await roleService.getUserRoles(user2.id);
 | 
			
		||||
			const actual3 = await roleService.getUserRoles(user3.id);
 | 
			
		||||
			expect(actual1.some(r => r.id === role.id)).toBe(true);
 | 
			
		||||
			expect(actual2.some(r => r.id === role.id)).toBe(true);
 | 
			
		||||
			expect(actual3.some(r => r.id === role.id)).toBe(false);
 | 
			
		||||
		});
 | 
			
		||||
 | 
			
		||||
		test('フォロー数が指定値以上', async () => {
 | 
			
		||||
			const [user1, user2, user3] = await Promise.all([
 | 
			
		||||
				createUser({ followingCount: 99 }),
 | 
			
		||||
				createUser({ followingCount: 100 }),
 | 
			
		||||
				createUser({ followingCount: 101 }),
 | 
			
		||||
			]);
 | 
			
		||||
			const role = await createConditionalRole({
 | 
			
		||||
				id: aidx(),
 | 
			
		||||
				type: 'followingMoreThanOrEq',
 | 
			
		||||
				value: 100,
 | 
			
		||||
			});
 | 
			
		||||
 | 
			
		||||
			const actual1 = await roleService.getUserRoles(user1.id);
 | 
			
		||||
			const actual2 = await roleService.getUserRoles(user2.id);
 | 
			
		||||
			const actual3 = await roleService.getUserRoles(user3.id);
 | 
			
		||||
			expect(actual1.some(r => r.id === role.id)).toBe(false);
 | 
			
		||||
			expect(actual2.some(r => r.id === role.id)).toBe(true);
 | 
			
		||||
			expect(actual3.some(r => r.id === role.id)).toBe(true);
 | 
			
		||||
		});
 | 
			
		||||
 | 
			
		||||
		test('ノート数が指定値以下', async () => {
 | 
			
		||||
			const [user1, user2, user3] = await Promise.all([
 | 
			
		||||
				createUser({ notesCount: 9 }),
 | 
			
		||||
				createUser({ notesCount: 10 }),
 | 
			
		||||
				createUser({ notesCount: 11 }),
 | 
			
		||||
			]);
 | 
			
		||||
			const role = await createConditionalRole({
 | 
			
		||||
				id: aidx(),
 | 
			
		||||
				type: 'notesLessThanOrEq',
 | 
			
		||||
				value: 10,
 | 
			
		||||
			});
 | 
			
		||||
 | 
			
		||||
			const actual1 = await roleService.getUserRoles(user1.id);
 | 
			
		||||
			const actual2 = await roleService.getUserRoles(user2.id);
 | 
			
		||||
			const actual3 = await roleService.getUserRoles(user3.id);
 | 
			
		||||
			expect(actual1.some(r => r.id === role.id)).toBe(true);
 | 
			
		||||
			expect(actual2.some(r => r.id === role.id)).toBe(true);
 | 
			
		||||
			expect(actual3.some(r => r.id === role.id)).toBe(false);
 | 
			
		||||
		});
 | 
			
		||||
 | 
			
		||||
		test('ノート数が指定値以上', async () => {
 | 
			
		||||
			const [user1, user2, user3] = await Promise.all([
 | 
			
		||||
				createUser({ notesCount: 9 }),
 | 
			
		||||
				createUser({ notesCount: 10 }),
 | 
			
		||||
				createUser({ notesCount: 11 }),
 | 
			
		||||
			]);
 | 
			
		||||
			const role = await createConditionalRole({
 | 
			
		||||
				id: aidx(),
 | 
			
		||||
				type: 'notesMoreThanOrEq',
 | 
			
		||||
				value: 10,
 | 
			
		||||
			});
 | 
			
		||||
 | 
			
		||||
			const actual1 = await roleService.getUserRoles(user1.id);
 | 
			
		||||
			const actual2 = await roleService.getUserRoles(user2.id);
 | 
			
		||||
			const actual3 = await roleService.getUserRoles(user3.id);
 | 
			
		||||
			expect(actual1.some(r => r.id === role.id)).toBe(false);
 | 
			
		||||
			expect(actual2.some(r => r.id === role.id)).toBe(true);
 | 
			
		||||
			expect(actual3.some(r => r.id === role.id)).toBe(true);
 | 
			
		||||
		});
 | 
			
		||||
	});
 | 
			
		||||
 | 
			
		||||
	describe('assign', () => {
 | 
			
		||||
		test('公開ロールの場合は通知される', async () => {
 | 
			
		||||
			const user = await createUser();
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										88
									
								
								packages/backend/test/unit/misc/is-renote.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										88
									
								
								packages/backend/test/unit/misc/is-renote.ts
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,88 @@
 | 
			
		|||
/*
 | 
			
		||||
 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
			
		||||
 * SPDX-License-Identifier: AGPL-3.0-only
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
import { isQuote, isRenote } from '@/misc/is-renote.js';
 | 
			
		||||
import { MiNote } from '@/models/Note.js';
 | 
			
		||||
 | 
			
		||||
const base: MiNote = {
 | 
			
		||||
	id: 'some-note-id',
 | 
			
		||||
	replyId: null,
 | 
			
		||||
	reply: null,
 | 
			
		||||
	renoteId: null,
 | 
			
		||||
	renote: null,
 | 
			
		||||
	threadId: null,
 | 
			
		||||
	text: null,
 | 
			
		||||
	name: null,
 | 
			
		||||
	cw: null,
 | 
			
		||||
	userId: 'some-user-id',
 | 
			
		||||
	user: null,
 | 
			
		||||
	localOnly: false,
 | 
			
		||||
	reactionAcceptance: null,
 | 
			
		||||
	renoteCount: 0,
 | 
			
		||||
	repliesCount: 0,
 | 
			
		||||
	clippedCount: 0,
 | 
			
		||||
	reactions: {},
 | 
			
		||||
	visibility: 'public',
 | 
			
		||||
	uri: null,
 | 
			
		||||
	url: null,
 | 
			
		||||
	fileIds: [],
 | 
			
		||||
	attachedFileTypes: [],
 | 
			
		||||
	visibleUserIds: [],
 | 
			
		||||
	mentions: [],
 | 
			
		||||
	mentionedRemoteUsers: '',
 | 
			
		||||
	reactionAndUserPairCache: [],
 | 
			
		||||
	emojis: [],
 | 
			
		||||
	tags: [],
 | 
			
		||||
	hasPoll: false,
 | 
			
		||||
	channelId: null,
 | 
			
		||||
	channel: null,
 | 
			
		||||
	userHost: null,
 | 
			
		||||
	replyUserId: null,
 | 
			
		||||
	replyUserHost: null,
 | 
			
		||||
	renoteUserId: null,
 | 
			
		||||
	renoteUserHost: null,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
describe('misc:is-renote', () => {
 | 
			
		||||
	test('note without renoteId should not be Renote', () => {
 | 
			
		||||
		expect(isRenote(base)).toBe(false);
 | 
			
		||||
	});
 | 
			
		||||
 | 
			
		||||
	test('note with renoteId should be Renote and not be Quote', () => {
 | 
			
		||||
		const note: MiNote = { ...base, renoteId: 'some-renote-id' };
 | 
			
		||||
		expect(isRenote(note)).toBe(true);
 | 
			
		||||
		expect(isQuote(note as any)).toBe(false);
 | 
			
		||||
	});
 | 
			
		||||
 | 
			
		||||
	test('note with renoteId and text should be Quote', () => {
 | 
			
		||||
		const note: MiNote = { ...base, renoteId: 'some-renote-id', text: 'some-text' };
 | 
			
		||||
		expect(isRenote(note)).toBe(true);
 | 
			
		||||
		expect(isQuote(note as any)).toBe(true);
 | 
			
		||||
	});
 | 
			
		||||
 | 
			
		||||
	test('note with renoteId and cw should be Quote', () => {
 | 
			
		||||
		const note: MiNote = { ...base, renoteId: 'some-renote-id', cw: 'some-cw' };
 | 
			
		||||
		expect(isRenote(note)).toBe(true);
 | 
			
		||||
		expect(isQuote(note as any)).toBe(true);
 | 
			
		||||
	});
 | 
			
		||||
 | 
			
		||||
	test('note with renoteId and replyId should be Quote', () => {
 | 
			
		||||
		const note: MiNote = { ...base, renoteId: 'some-renote-id', replyId: 'some-reply-id' };
 | 
			
		||||
		expect(isRenote(note)).toBe(true);
 | 
			
		||||
		expect(isQuote(note as any)).toBe(true);
 | 
			
		||||
	});
 | 
			
		||||
 | 
			
		||||
	test('note with renoteId and poll should be Quote', () => {
 | 
			
		||||
		const note: MiNote = { ...base, renoteId: 'some-renote-id', hasPoll: true };
 | 
			
		||||
		expect(isRenote(note)).toBe(true);
 | 
			
		||||
		expect(isQuote(note as any)).toBe(true);
 | 
			
		||||
	});
 | 
			
		||||
 | 
			
		||||
	test('note with renoteId and non-empty fileIds should be Quote', () => {
 | 
			
		||||
		const note: MiNote = { ...base, renoteId: 'some-renote-id', fileIds: ['some-file-id'] };
 | 
			
		||||
		expect(isRenote(note)).toBe(true);
 | 
			
		||||
		expect(isQuote(note as any)).toBe(true);
 | 
			
		||||
	});
 | 
			
		||||
});
 | 
			
		||||
| 
						 | 
				
			
			@ -25,7 +25,7 @@
 | 
			
		|||
		"@rollup/plugin-replace": "5.0.5",
 | 
			
		||||
		"@rollup/pluginutils": "5.1.0",
 | 
			
		||||
		"@transfem-org/sfm-js": "0.24.4",
 | 
			
		||||
		"@syuilo/aiscript": "0.17.0",
 | 
			
		||||
		"@syuilo/aiscript": "0.18.0",
 | 
			
		||||
		"@phosphor-icons/web": "^2.0.3",
 | 
			
		||||
		"@twemoji/parser": "15.0.0",
 | 
			
		||||
		"@vitejs/plugin-vue": "5.0.4",
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -4,37 +4,59 @@ SPDX-License-Identifier: AGPL-3.0-only
 | 
			
		|||
-->
 | 
			
		||||
 | 
			
		||||
<template>
 | 
			
		||||
<div :class="$style.root" class="_panel">
 | 
			
		||||
	<b>{{ clip.name }}</b>
 | 
			
		||||
	<div v-if="clip.description" :class="$style.description">{{ clip.description }}</div>
 | 
			
		||||
	<div v-if="clip.lastClippedAt">{{ i18n.ts.updatedAt }}: <MkTime :time="clip.lastClippedAt" mode="detail"/></div>
 | 
			
		||||
	<div :class="$style.user">
 | 
			
		||||
		<MkAvatar :user="clip.user" :class="$style.userAvatar" indicator link preview/> <MkUserName :user="clip.user" :nowrap="false"/>
 | 
			
		||||
<MkA :to="`/clips/${clip.id}`" :class="$style.link">
 | 
			
		||||
	<div :class="$style.root" class="_panel _gaps_s">
 | 
			
		||||
		<b>{{ clip.name }}</b>
 | 
			
		||||
		<div :class="$style.description">
 | 
			
		||||
			<div v-if="clip.description"><Mfm :text="clip.description" :plain="true" :nowrap="true"/></div>
 | 
			
		||||
			<div v-if="clip.lastClippedAt">{{ i18n.ts.updatedAt }}: <MkTime :time="clip.lastClippedAt" mode="detail"/></div>
 | 
			
		||||
			<div v-if="clip.notesCount != null">{{ i18n.ts.notesCount }}: {{ number(clip.notesCount) }} / {{ $i?.policies.noteEachClipsLimit }} ({{ i18n.tsx.remainingN({ n: remaining }) }})</div>
 | 
			
		||||
		</div>
 | 
			
		||||
		<div :class="$style.divider"></div>
 | 
			
		||||
		<div>
 | 
			
		||||
			<MkAvatar :user="clip.user" :class="$style.userAvatar" indicator link preview/> <MkUserName :user="clip.user" :nowrap="false"/>
 | 
			
		||||
		</div>
 | 
			
		||||
	</div>
 | 
			
		||||
</div>
 | 
			
		||||
</MkA>
 | 
			
		||||
</template>
 | 
			
		||||
 | 
			
		||||
<script lang="ts" setup>
 | 
			
		||||
import * as Misskey from 'misskey-js';
 | 
			
		||||
import { computed } from 'vue';
 | 
			
		||||
import { i18n } from '@/i18n.js';
 | 
			
		||||
import { $i } from '@/account.js';
 | 
			
		||||
import number from '@/filters/number.js';
 | 
			
		||||
 | 
			
		||||
defineProps<{
 | 
			
		||||
	clip: any;
 | 
			
		||||
const props = defineProps<{
 | 
			
		||||
	clip: Misskey.entities.Clip;
 | 
			
		||||
}>();
 | 
			
		||||
 | 
			
		||||
const remaining = computed(() => {
 | 
			
		||||
	return ($i?.policies && props.clip.notesCount != null) ? ($i.policies.noteEachClipsLimit - props.clip.notesCount) : i18n.ts.unknown;
 | 
			
		||||
});
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
<style lang="scss" module>
 | 
			
		||||
.root {
 | 
			
		||||
.link {
 | 
			
		||||
	display: block;
 | 
			
		||||
 | 
			
		||||
	&:hover {
 | 
			
		||||
		text-decoration: none;
 | 
			
		||||
		color: var(--accent);
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.root {
 | 
			
		||||
	padding: 16px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.description {
 | 
			
		||||
	padding: 8px 0;
 | 
			
		||||
.divider {
 | 
			
		||||
	height: 1px;
 | 
			
		||||
	background: var(--divider);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.user {
 | 
			
		||||
	padding-top: 16px;
 | 
			
		||||
	border-top: solid 0.5px var(--divider);
 | 
			
		||||
.description {
 | 
			
		||||
	font-size: 90%;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.userAvatar {
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -93,6 +93,18 @@ async function onClick() {
 | 
			
		|||
				userId: props.user.id,
 | 
			
		||||
			});
 | 
			
		||||
		} else {
 | 
			
		||||
			if (defaultStore.state.alwaysConfirmFollow) {
 | 
			
		||||
				const { canceled } = await os.confirm({
 | 
			
		||||
					type: 'question',
 | 
			
		||||
					text: i18n.tsx.followConfirm({ name: props.user.name || props.user.username }),
 | 
			
		||||
				});
 | 
			
		||||
 | 
			
		||||
				if (canceled) {
 | 
			
		||||
					wait.value = false;
 | 
			
		||||
					return;
 | 
			
		||||
				}
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			if (hasPendingFollowRequestFromYou.value) {
 | 
			
		||||
				await misskeyApi('following/requests/cancel', {
 | 
			
		||||
					userId: props.user.id,
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -394,7 +394,7 @@ function addMissingMention() {
 | 
			
		|||
	for (const x of extractMentions(ast)) {
 | 
			
		||||
		if (!visibleUsers.value.some(u => (u.username === x.username) && (u.host === x.host))) {
 | 
			
		||||
			misskeyApi('users/show', { username: x.username, host: x.host }).then(user => {
 | 
			
		||||
				visibleUsers.value.push(user);
 | 
			
		||||
				pushVisibleUser(user);
 | 
			
		||||
			});
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
| 
						 | 
				
			
			@ -685,6 +685,7 @@ function saveDraft() {
 | 
			
		|||
			localOnly: localOnly.value,
 | 
			
		||||
			files: files.value,
 | 
			
		||||
			poll: poll.value,
 | 
			
		||||
			visibleUserIds: visibility.value === 'specified' ? visibleUsers.value.map(x => x.id) : undefined,
 | 
			
		||||
		},
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -997,6 +998,15 @@ onMounted(() => {
 | 
			
		|||
				if (draft.data.poll) {
 | 
			
		||||
					poll.value = draft.data.poll;
 | 
			
		||||
				}
 | 
			
		||||
				if (draft.data.visibleUserIds) {
 | 
			
		||||
					misskeyApi('users/show', { userIds: draft.data.visibleUserIds }).then(users => {
 | 
			
		||||
						for (let i = 0; i < users.length; i++) {
 | 
			
		||||
							if (users[i].id === draft.data.visibleUserIds[i]) {
 | 
			
		||||
								pushVisibleUser(users[i]);
 | 
			
		||||
							}
 | 
			
		||||
						}
 | 
			
		||||
					});
 | 
			
		||||
				}
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -9,6 +9,11 @@ SPDX-License-Identifier: AGPL-3.0-only
 | 
			
		|||
		<MkSelect v-model="type" :class="$style.typeSelect">
 | 
			
		||||
			<option value="isLocal">{{ i18n.ts._role._condition.isLocal }}</option>
 | 
			
		||||
			<option value="isRemote">{{ i18n.ts._role._condition.isRemote }}</option>
 | 
			
		||||
			<option value="isSuspended">{{ i18n.ts._role._condition.isSuspended }}</option>
 | 
			
		||||
			<option value="isLocked">{{ i18n.ts._role._condition.isLocked }}</option>
 | 
			
		||||
			<option value="isBot">{{ i18n.ts._role._condition.isBot }}</option>
 | 
			
		||||
			<option value="isCat">{{ i18n.ts._role._condition.isCat }}</option>
 | 
			
		||||
			<option value="isExplorable">{{ i18n.ts._role._condition.isExplorable }}</option>
 | 
			
		||||
			<option value="roleAssignedTo">{{ i18n.ts._role._condition.roleAssignedTo }}</option>
 | 
			
		||||
			<option value="createdLessThan">{{ i18n.ts._role._condition.createdLessThan }}</option>
 | 
			
		||||
			<option value="createdMoreThan">{{ i18n.ts._role._condition.createdMoreThan }}</option>
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -9,11 +9,16 @@ SPDX-License-Identifier: AGPL-3.0-only
 | 
			
		|||
	<MkSpacer :contentMax="800">
 | 
			
		||||
		<div v-if="clip" class="_gaps">
 | 
			
		||||
			<div class="_panel">
 | 
			
		||||
				<div v-if="clip.description" :class="$style.description">
 | 
			
		||||
					<Mfm :text="clip.description" :isNote="false"/>
 | 
			
		||||
				<div class="_gaps_s" :class="$style.description">
 | 
			
		||||
					<div v-if="clip.description">
 | 
			
		||||
						<Mfm :text="clip.description" :isNote="false"/>
 | 
			
		||||
					</div>
 | 
			
		||||
					<div v-else>({{ i18n.ts.noDescription }})</div>
 | 
			
		||||
					<div>
 | 
			
		||||
						<MkButton v-if="favorited" v-tooltip="i18n.ts.unfavorite" asLike rounded primary @click="unfavorite()"><i class="ti ti-heart"></i><span v-if="clip.favoritedCount > 0" style="margin-left: 6px;">{{ clip.favoritedCount }}</span></MkButton>
 | 
			
		||||
						<MkButton v-else v-tooltip="i18n.ts.favorite" asLike rounded @click="favorite()"><i class="ti ti-heart"></i><span v-if="clip.favoritedCount > 0" style="margin-left: 6px;">{{ clip.favoritedCount }}</span></MkButton>
 | 
			
		||||
					</div>
 | 
			
		||||
				</div>
 | 
			
		||||
				<MkButton v-if="favorited" v-tooltip="i18n.ts.unfavorite" asLike rounded primary @click="unfavorite()"><i class="ph-heart ph-bold ph-lg"></i><span v-if="clip.favoritedCount > 0" style="margin-left: 6px;">{{ clip.favoritedCount }}</span></MkButton>
 | 
			
		||||
				<MkButton v-else v-tooltip="i18n.ts.favorite" asLike rounded @click="favorite()"><i class="ph-heart ph-bold ph-lg"></i><span v-if="clip.favoritedCount > 0" style="margin-left: 6px;">{{ clip.favoritedCount }}</span></MkButton>
 | 
			
		||||
				<div :class="$style.user">
 | 
			
		||||
					<MkAvatar :user="clip.user" :class="$style.avatar" indicator link preview/> <MkUserName :user="clip.user" :nowrap="false"/>
 | 
			
		||||
				</div>
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -48,7 +48,7 @@ import MkInput from '@/components/MkInput.vue';
 | 
			
		|||
import MkSelect from '@/components/MkSelect.vue';
 | 
			
		||||
import { useRouter } from '@/router/supplier.js';
 | 
			
		||||
 | 
			
		||||
const PRESET_DEFAULT = `/// @ 0.16.0
 | 
			
		||||
const PRESET_DEFAULT = `/// @ 0.18.0
 | 
			
		||||
 | 
			
		||||
var name = ""
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -60,13 +60,13 @@ Ui:render([
 | 
			
		|||
	Ui:C:button({
 | 
			
		||||
		text: "Hello"
 | 
			
		||||
		onClick: @() {
 | 
			
		||||
			Mk:dialog(null \`Hello, {name}!\`)
 | 
			
		||||
			Mk:dialog(null, \`Hello, {name}!\`)
 | 
			
		||||
		}
 | 
			
		||||
	})
 | 
			
		||||
])
 | 
			
		||||
`;
 | 
			
		||||
 | 
			
		||||
const PRESET_OMIKUJI = `/// @ 0.16.0
 | 
			
		||||
const PRESET_OMIKUJI = `/// @ 0.18.0
 | 
			
		||||
// ユーザーごとに日替わりのおみくじのプリセット
 | 
			
		||||
 | 
			
		||||
// 選択肢
 | 
			
		||||
| 
						 | 
				
			
			@ -81,11 +81,11 @@ let choices = [
 | 
			
		|||
	"大凶"
 | 
			
		||||
]
 | 
			
		||||
 | 
			
		||||
// シードが「ユーザーID+今日の日付」である乱数生成器を用意
 | 
			
		||||
let random = Math:gen_rng(\`{USER_ID}{Date:year()}{Date:month()}{Date:day()}\`)
 | 
			
		||||
// シードが「PlayID+ユーザーID+今日の日付」である乱数生成器を用意
 | 
			
		||||
let random = Math:gen_rng(\`{THIS_ID}{USER_ID}{Date:year()}{Date:month()}{Date:day()}\`)
 | 
			
		||||
 | 
			
		||||
// ランダムに選択肢を選ぶ
 | 
			
		||||
let chosen = choices[random(0 (choices.len - 1))]
 | 
			
		||||
let chosen = choices[random(0, (choices.len - 1))]
 | 
			
		||||
 | 
			
		||||
// 結果のテキスト
 | 
			
		||||
let result = \`今日のあなたの運勢は **{chosen}** です。\`
 | 
			
		||||
| 
						 | 
				
			
			@ -109,7 +109,7 @@ Ui:render([
 | 
			
		|||
])
 | 
			
		||||
`;
 | 
			
		||||
 | 
			
		||||
const PRESET_SHUFFLE = `/// @ 0.16.0
 | 
			
		||||
const PRESET_SHUFFLE = `/// @ 0.18.0
 | 
			
		||||
// 巻き戻し可能な文字シャッフルのプリセット
 | 
			
		||||
 | 
			
		||||
let string = "ペペロンチーノ"
 | 
			
		||||
| 
						 | 
				
			
			@ -123,13 +123,13 @@ var cursor = 0
 | 
			
		|||
 | 
			
		||||
@do() {
 | 
			
		||||
	if (cursor != 0) {
 | 
			
		||||
		results = results.slice(0 (cursor + 1))
 | 
			
		||||
		results = results.slice(0, (cursor + 1))
 | 
			
		||||
		cursor = 0
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	let chars = []
 | 
			
		||||
	for (let i, length) {
 | 
			
		||||
		let r = Math:rnd(0 (length - 1))
 | 
			
		||||
		let r = Math:rnd(0, (length - 1))
 | 
			
		||||
		chars.push(string.pick(r))
 | 
			
		||||
	}
 | 
			
		||||
	let result = chars.join("")
 | 
			
		||||
| 
						 | 
				
			
			@ -188,27 +188,27 @@ var cursor = 0
 | 
			
		|||
do()
 | 
			
		||||
`;
 | 
			
		||||
 | 
			
		||||
const PRESET_QUIZ = `/// @ 0.16.0
 | 
			
		||||
const PRESET_QUIZ = `/// @ 0.18.0
 | 
			
		||||
let title = '地理クイズ'
 | 
			
		||||
 | 
			
		||||
let qas = [{
 | 
			
		||||
	q: 'オーストラリアの首都は?'
 | 
			
		||||
	choices: ['シドニー' 'キャンベラ' 'メルボルン']
 | 
			
		||||
	choices: ['シドニー', 'キャンベラ', 'メルボルン']
 | 
			
		||||
	a: 'キャンベラ'
 | 
			
		||||
	aDescription: '最大の都市はシドニーですが首都はキャンベラです。'
 | 
			
		||||
} {
 | 
			
		||||
	q: '国土面積2番目の国は?'
 | 
			
		||||
	choices: ['カナダ' 'アメリカ' '中国']
 | 
			
		||||
	choices: ['カナダ', 'アメリカ', '中国']
 | 
			
		||||
	a: 'カナダ'
 | 
			
		||||
	aDescription: '大きい順にロシア、カナダ、アメリカ、中国です。'
 | 
			
		||||
} {
 | 
			
		||||
	q: '二重内陸国ではないのは?'
 | 
			
		||||
	choices: ['リヒテンシュタイン' 'ウズベキスタン' 'レソト']
 | 
			
		||||
	choices: ['リヒテンシュタイン', 'ウズベキスタン', 'レソト']
 | 
			
		||||
	a: 'レソト'
 | 
			
		||||
	aDescription: 'レソトは(一重)内陸国です。'
 | 
			
		||||
} {
 | 
			
		||||
	q: '閘門がない運河は?'
 | 
			
		||||
	choices: ['キール運河' 'スエズ運河' 'パナマ運河']
 | 
			
		||||
	choices: ['キール運河', 'スエズ運河', 'パナマ運河']
 | 
			
		||||
	a: 'スエズ運河'
 | 
			
		||||
	aDescription: 'スエズ運河は高低差がないので閘門はありません。'
 | 
			
		||||
}]
 | 
			
		||||
| 
						 | 
				
			
			@ -296,12 +296,12 @@ qaEls.push(Ui:C:container({
 | 
			
		|||
			onClick: finish
 | 
			
		||||
		})
 | 
			
		||||
	]
 | 
			
		||||
} 'footer'))
 | 
			
		||||
}, 'footer'))
 | 
			
		||||
 | 
			
		||||
Ui:render(qaEls)
 | 
			
		||||
`;
 | 
			
		||||
 | 
			
		||||
const PRESET_TIMELINE = `/// @ 0.16.0
 | 
			
		||||
const PRESET_TIMELINE = `/// @ 0.18.0
 | 
			
		||||
// APIリクエストを行いローカルタイムラインを表示するプリセット
 | 
			
		||||
 | 
			
		||||
@fetch() {
 | 
			
		||||
| 
						 | 
				
			
			@ -315,7 +315,7 @@ const PRESET_TIMELINE = `/// @ 0.16.0
 | 
			
		|||
	])
 | 
			
		||||
 | 
			
		||||
	// タイムライン取得
 | 
			
		||||
	let notes = Mk:api("notes/local-timeline" {})
 | 
			
		||||
	let notes = Mk:api("notes/local-timeline", {})
 | 
			
		||||
 | 
			
		||||
	// それぞれのノートごとにUI要素作成
 | 
			
		||||
	let noteEls = []
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -11,16 +11,12 @@ SPDX-License-Identifier: AGPL-3.0-only
 | 
			
		|||
			<div v-if="tab === 'my'" key="my" class="_gaps">
 | 
			
		||||
				<MkButton primary rounded class="add" @click="create"><i class="ph-plus ph-bold ph-lg"></i> {{ i18n.ts.add }}</MkButton>
 | 
			
		||||
 | 
			
		||||
				<MkPagination v-slot="{items}" ref="pagingComponent" :pagination="pagination" class="_gaps">
 | 
			
		||||
					<MkA v-for="item in items" :key="item.id" :to="`/clips/${item.id}`">
 | 
			
		||||
						<MkClipPreview :clip="item"/>
 | 
			
		||||
					</MkA>
 | 
			
		||||
				<MkPagination v-slot="{ items }" ref="pagingComponent" :pagination="pagination" class="_gaps">
 | 
			
		||||
					<MkClipPreview v-for="item in items" :key="item.id" :clip="item"/>
 | 
			
		||||
				</MkPagination>
 | 
			
		||||
			</div>
 | 
			
		||||
			<div v-else-if="tab === 'favorites'" key="favorites" class="_gaps">
 | 
			
		||||
				<MkA v-for="item in favorites" :key="item.id" :to="`/clips/${item.id}`">
 | 
			
		||||
					<MkClipPreview :clip="item"/>
 | 
			
		||||
				</MkA>
 | 
			
		||||
				<MkClipPreview v-for="item in favorites" :key="item.id" :clip="item"/>
 | 
			
		||||
			</div>
 | 
			
		||||
		</MkHorizontalSwipe>
 | 
			
		||||
	</MkSpacer>
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -30,9 +30,7 @@ SPDX-License-Identifier: AGPL-3.0-only
 | 
			
		|||
						<div v-if="clips && clips.length > 0" class="_margin">
 | 
			
		||||
							<div style="font-weight: bold; padding: 12px;">{{ i18n.ts.clip }}</div>
 | 
			
		||||
							<div class="_gaps">
 | 
			
		||||
								<MkA v-for="item in clips" :key="item.id" :to="`/clips/${item.id}`">
 | 
			
		||||
									<MkClipPreview :clip="item"/>
 | 
			
		||||
								</MkA>
 | 
			
		||||
								<MkClipPreview v-for="item in clips" :key="item.id" :clip="item"/>
 | 
			
		||||
							</div>
 | 
			
		||||
						</div>
 | 
			
		||||
						<div v-if="!showPrev" class="_buttons" :class="$style.loadPrev">
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -12,6 +12,7 @@ SPDX-License-Identifier: AGPL-3.0-only
 | 
			
		|||
			:leaveActiveClass="defaultStore.state.animation ? $style.fadeLeaveActive : ''"
 | 
			
		||||
			:enterFromClass="defaultStore.state.animation ? $style.fadeEnterFrom : ''"
 | 
			
		||||
			:leaveToClass="defaultStore.state.animation ? $style.fadeLeaveTo : ''"
 | 
			
		||||
			mode="out-in"
 | 
			
		||||
		>
 | 
			
		||||
			<div v-if="page" :key="page.id" class="_gaps">
 | 
			
		||||
				<div :class="$style.pageMain">
 | 
			
		||||
| 
						 | 
				
			
			@ -41,8 +42,14 @@ SPDX-License-Identifier: AGPL-3.0-only
 | 
			
		|||
						</div>
 | 
			
		||||
						<div :class="$style.pageBannerTitle" class="_gaps_s">
 | 
			
		||||
							<h1>{{ page.title || page.name }}</h1>
 | 
			
		||||
							<div v-if="page.user" :class="$style.pageBannerTitleUser">
 | 
			
		||||
								<MkAvatar :user="page.user" :class="$style.avatar" indicator link preview/> <MkA :to="`/@${username}`"><MkUserName :user="page.user" :nowrap="false"/></MkA>
 | 
			
		||||
							<div :class="$style.pageBannerTitleSub">
 | 
			
		||||
								<div v-if="page.user" :class="$style.pageBannerTitleUser">
 | 
			
		||||
									<MkAvatar :user="page.user" :class="$style.avatar" indicator link preview/> <MkA :to="`/@${username}`"><MkUserName :user="page.user" :nowrap="false"/></MkA>
 | 
			
		||||
								</div>
 | 
			
		||||
								<div :class="$style.pageBannerTitleSubActions">
 | 
			
		||||
									<MkA v-if="page.userId === $i?.id" v-tooltip="i18n.ts._pages.editThisPage" :to="`/pages/edit/${page.id}`" class="_button" :class="$style.generalActionButton"><i class="ti ti-pencil ti-fw"></i></MkA>
 | 
			
		||||
									<button v-tooltip="i18n.ts.share" class="_button" :class="$style.generalActionButton" @click="share"><i class="ti ti-share ti-fw"></i></button>
 | 
			
		||||
								</div>
 | 
			
		||||
							</div>
 | 
			
		||||
						</div>
 | 
			
		||||
					</div>
 | 
			
		||||
| 
						 | 
				
			
			@ -355,8 +362,15 @@ definePageMetadata(() => ({
 | 
			
		|||
			margin: 0;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		.pageBannerTitleSub {
 | 
			
		||||
			display: flex;
 | 
			
		||||
			align-items: center;
 | 
			
		||||
			width: 100%;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		.pageBannerTitleUser {
 | 
			
		||||
			--height: 32px;
 | 
			
		||||
			flex-shrink: 0;
 | 
			
		||||
 | 
			
		||||
			.avatar {
 | 
			
		||||
				height: var(--height);
 | 
			
		||||
| 
						 | 
				
			
			@ -365,6 +379,14 @@ definePageMetadata(() => ({
 | 
			
		|||
 | 
			
		||||
			line-height: var(--height);
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		.pageBannerTitleSubActions {
 | 
			
		||||
			flex-shrink: 0;
 | 
			
		||||
			display: flex;
 | 
			
		||||
			align-items: center;
 | 
			
		||||
			gap: var(--marginHalf);
 | 
			
		||||
			margin-left: auto;
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -44,6 +44,10 @@ SPDX-License-Identifier: AGPL-3.0-only
 | 
			
		|||
				<template #label>{{ i18n.ts.keepOriginalUploading }}</template>
 | 
			
		||||
				<template #caption>{{ i18n.ts.keepOriginalUploadingDescription }}</template>
 | 
			
		||||
			</MkSwitch>
 | 
			
		||||
			<MkSwitch v-model="keepOriginalFilename">
 | 
			
		||||
				<template #label>{{ i18n.ts.keepOriginalFilename }}</template>
 | 
			
		||||
				<template #caption>{{ i18n.ts.keepOriginalFilenameDescription }}</template>
 | 
			
		||||
			</MkSwitch>
 | 
			
		||||
			<MkSwitch v-model="alwaysMarkNsfw" @update:modelValue="saveProfile()">
 | 
			
		||||
				<template #label>{{ i18n.ts.alwaysMarkSensitive }}</template>
 | 
			
		||||
			</MkSwitch>
 | 
			
		||||
| 
						 | 
				
			
			@ -91,6 +95,7 @@ const meterStyle = computed(() => {
 | 
			
		|||
});
 | 
			
		||||
 | 
			
		||||
const keepOriginalUploading = computed(defaultStore.makeGetterSetter('keepOriginalUploading'));
 | 
			
		||||
const keepOriginalFilename = computed(defaultStore.makeGetterSetter('keepOriginalFilename'));
 | 
			
		||||
 | 
			
		||||
misskeyApi('drive').then(info => {
 | 
			
		||||
	capacity.value = info.capacity;
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -191,6 +191,7 @@ SPDX-License-Identifier: AGPL-3.0-only
 | 
			
		|||
				<MkSwitch v-model="showBots">{{ i18n.ts.showBots }}</MkSwitch>
 | 
			
		||||
				<MkSwitch v-model="disableStreamingTimeline">{{ i18n.ts.disableStreamingTimeline }}</MkSwitch>
 | 
			
		||||
				<MkSwitch v-model="enableHorizontalSwipe">{{ i18n.ts.enableHorizontalSwipe }}</MkSwitch>
 | 
			
		||||
				<MkSwitch v-model="alwaysConfirmFollow">{{ i18n.ts.alwaysConfirmFollow }}</MkSwitch>
 | 
			
		||||
			</div>
 | 
			
		||||
			<MkSelect v-model="serverDisconnectedBehavior">
 | 
			
		||||
				<template #label>{{ i18n.ts.whenServerDisconnected }}</template>
 | 
			
		||||
| 
						 | 
				
			
			@ -366,6 +367,7 @@ const showVisibilitySelectorOnBoost = computed(defaultStore.makeGetterSetter('sh
 | 
			
		|||
const visibilityOnBoost = computed(defaultStore.makeGetterSetter('visibilityOnBoost'));
 | 
			
		||||
const enableHorizontalSwipe = computed(defaultStore.makeGetterSetter('enableHorizontalSwipe'));
 | 
			
		||||
const useNativeUIForVideoAudioPlayer = computed(defaultStore.makeGetterSetter('useNativeUIForVideoAudioPlayer'));
 | 
			
		||||
const alwaysConfirmFollow = computed(defaultStore.makeGetterSetter('alwaysConfirmFollow'));
 | 
			
		||||
 | 
			
		||||
watch(lang, () => {
 | 
			
		||||
	miLocalStorage.setItem('lang', lang.value as string);
 | 
			
		||||
| 
						 | 
				
			
			@ -424,6 +426,7 @@ watch([
 | 
			
		|||
	enableSeasonalScreenEffect,
 | 
			
		||||
	showVisibilitySelectorOnBoost,
 | 
			
		||||
	visibilityOnBoost,
 | 
			
		||||
	alwaysConfirmFollow,
 | 
			
		||||
], async () => {
 | 
			
		||||
	await reloadAsk();
 | 
			
		||||
});
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -26,6 +26,14 @@ export async function getNoteClipMenu(props: {
 | 
			
		|||
	isDeleted: Ref<boolean>;
 | 
			
		||||
	currentClip?: Misskey.entities.Clip;
 | 
			
		||||
}) {
 | 
			
		||||
	function getClipName(clip: Misskey.entities.Clip) {
 | 
			
		||||
		if ($i && clip.userId === $i.id && clip.notesCount != null) {
 | 
			
		||||
			return `${clip.name} (${clip.notesCount}/${$i.policies.noteEachClipsLimit})`;
 | 
			
		||||
		} else {
 | 
			
		||||
			return clip.name;
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	const isRenote = (
 | 
			
		||||
		props.note.renote != null &&
 | 
			
		||||
		props.note.text == null &&
 | 
			
		||||
| 
						 | 
				
			
			@ -37,7 +45,7 @@ export async function getNoteClipMenu(props: {
 | 
			
		|||
 | 
			
		||||
	const clips = await clipsCache.fetch();
 | 
			
		||||
	const menu: MenuItem[] = [...clips.map(clip => ({
 | 
			
		||||
		text: clip.name,
 | 
			
		||||
		text: getClipName(clip),
 | 
			
		||||
		action: () => {
 | 
			
		||||
			claimAchievement('noteClipped1');
 | 
			
		||||
			os.promiseDialog(
 | 
			
		||||
| 
						 | 
				
			
			@ -50,7 +58,18 @@ export async function getNoteClipMenu(props: {
 | 
			
		|||
							text: i18n.tsx.confirmToUnclipAlreadyClippedNote({ name: clip.name }),
 | 
			
		||||
						});
 | 
			
		||||
						if (!confirm.canceled) {
 | 
			
		||||
							os.apiWithDialog('clips/remove-note', { clipId: clip.id, noteId: appearNote.id });
 | 
			
		||||
							os.apiWithDialog('clips/remove-note', { clipId: clip.id, noteId: appearNote.id }).then(() => {
 | 
			
		||||
								clipsCache.set(clips.map(c => {
 | 
			
		||||
									if (c.id === clip.id) {
 | 
			
		||||
										return {
 | 
			
		||||
											...c,
 | 
			
		||||
											notesCount: Math.max(0, ((c.notesCount ?? 0) - 1)),
 | 
			
		||||
										};
 | 
			
		||||
									} else {
 | 
			
		||||
										return c;
 | 
			
		||||
									}
 | 
			
		||||
								}));
 | 
			
		||||
							});
 | 
			
		||||
							if (props.currentClip?.id === clip.id) props.isDeleted.value = true;
 | 
			
		||||
						}
 | 
			
		||||
					} else {
 | 
			
		||||
| 
						 | 
				
			
			@ -60,7 +79,18 @@ export async function getNoteClipMenu(props: {
 | 
			
		|||
						});
 | 
			
		||||
					}
 | 
			
		||||
				},
 | 
			
		||||
			);
 | 
			
		||||
			).then(() => {
 | 
			
		||||
				clipsCache.set(clips.map(c => {
 | 
			
		||||
					if (c.id === clip.id) {
 | 
			
		||||
						return {
 | 
			
		||||
							...c,
 | 
			
		||||
							notesCount: (c.notesCount ?? 0) + 1,
 | 
			
		||||
						};
 | 
			
		||||
					} else {
 | 
			
		||||
						return c;
 | 
			
		||||
					}
 | 
			
		||||
				}));
 | 
			
		||||
			});
 | 
			
		||||
		},
 | 
			
		||||
	})), { type: 'divider' }, {
 | 
			
		||||
		icon: 'ph-plus ph-bold ph-lg',
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -15,6 +15,16 @@ const fallbackName = (key: string) => `idbfallback::${key}`;
 | 
			
		|||
 | 
			
		||||
let idbAvailable = typeof window !== 'undefined' ? !!(window.indexedDB && window.indexedDB.open) : true;
 | 
			
		||||
 | 
			
		||||
// iframe.contentWindow.indexedDB.deleteDatabase() がchromeのバグで使用できないため、indexedDBを無効化している。
 | 
			
		||||
// バグが治って再度有効化するのであれば、cypressのコマンド内のコメントアウトを外すこと
 | 
			
		||||
// see https://github.com/misskey-dev/misskey/issues/13605#issuecomment-2053652123
 | 
			
		||||
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
 | 
			
		||||
// @ts-expect-error
 | 
			
		||||
if (window.Cypress) {
 | 
			
		||||
	idbAvailable = false;
 | 
			
		||||
	console.log('Cypress detected. It will use localStorage.');
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
if (idbAvailable) {
 | 
			
		||||
	await iset('idb-test', 'test')
 | 
			
		||||
		.catch(err => {
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -5,6 +5,7 @@
 | 
			
		|||
 | 
			
		||||
import { reactive, ref } from 'vue';
 | 
			
		||||
import * as Misskey from 'misskey-js';
 | 
			
		||||
import { v4 as uuid } from 'uuid';
 | 
			
		||||
import { readAndCompressImage } from '@misskey-dev/browser-image-resizer';
 | 
			
		||||
import { getCompressionConfig } from './upload/compress-config.js';
 | 
			
		||||
import { defaultStore } from '@/store.js';
 | 
			
		||||
| 
						 | 
				
			
			@ -39,13 +40,16 @@ export function uploadFile(
 | 
			
		|||
	if (folder && typeof folder === 'object') folder = folder.id;
 | 
			
		||||
 | 
			
		||||
	return new Promise((resolve, reject) => {
 | 
			
		||||
		const id = Math.random().toString();
 | 
			
		||||
		const id = uuid();
 | 
			
		||||
 | 
			
		||||
		const reader = new FileReader();
 | 
			
		||||
		reader.onload = async (): Promise<void> => {
 | 
			
		||||
			const filename = name ?? file.name ?? 'untitled';
 | 
			
		||||
			const extension = filename.split('.').length > 1 ? '.' + filename.split('.').pop() : '';
 | 
			
		||||
 | 
			
		||||
			const ctx = reactive<Uploading>({
 | 
			
		||||
				id: id,
 | 
			
		||||
				name: name ?? file.name ?? 'untitled',
 | 
			
		||||
				id,
 | 
			
		||||
				name: defaultStore.state.keepOriginalFilename ? filename : id + extension,
 | 
			
		||||
				progressMax: undefined,
 | 
			
		||||
				progressValue: undefined,
 | 
			
		||||
				img: window.URL.createObjectURL(file),
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -499,6 +499,14 @@ export const defaultStore = markRaw(new Storage('base', {
 | 
			
		|||
		where: 'device',
 | 
			
		||||
		default: false,
 | 
			
		||||
	},
 | 
			
		||||
	keepOriginalFilename: {
 | 
			
		||||
		where: 'device',
 | 
			
		||||
		default: true,
 | 
			
		||||
	},
 | 
			
		||||
	alwaysConfirmFollow: {
 | 
			
		||||
		where: 'device',
 | 
			
		||||
		default: true,
 | 
			
		||||
	},
 | 
			
		||||
 | 
			
		||||
	sound_masterVolume: {
 | 
			
		||||
		where: 'device',
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1759,6 +1759,7 @@ declare namespace entities {
 | 
			
		|||
        RoleCondFormulaLogics,
 | 
			
		||||
        RoleCondFormulaValueNot,
 | 
			
		||||
        RoleCondFormulaValueIsLocalOrRemote,
 | 
			
		||||
        RoleCondFormulaValueUserSettingBooleanSchema,
 | 
			
		||||
        RoleCondFormulaValueAssignedRole,
 | 
			
		||||
        RoleCondFormulaValueCreated,
 | 
			
		||||
        RoleCondFormulaFollowersOrFollowingOrNotes,
 | 
			
		||||
| 
						 | 
				
			
			@ -2821,6 +2822,9 @@ type RoleCondFormulaValueIsLocalOrRemote = components['schemas']['RoleCondFormul
 | 
			
		|||
// @public (undocumented)
 | 
			
		||||
type RoleCondFormulaValueNot = components['schemas']['RoleCondFormulaValueNot'];
 | 
			
		||||
 | 
			
		||||
// @public (undocumented)
 | 
			
		||||
type RoleCondFormulaValueUserSettingBooleanSchema = components['schemas']['RoleCondFormulaValueUserSettingBooleanSchema'];
 | 
			
		||||
 | 
			
		||||
// @public (undocumented)
 | 
			
		||||
type RoleLite = components['schemas']['RoleLite'];
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -38,6 +38,7 @@ export type Signin = components['schemas']['Signin'];
 | 
			
		|||
export type RoleCondFormulaLogics = components['schemas']['RoleCondFormulaLogics'];
 | 
			
		||||
export type RoleCondFormulaValueNot = components['schemas']['RoleCondFormulaValueNot'];
 | 
			
		||||
export type RoleCondFormulaValueIsLocalOrRemote = components['schemas']['RoleCondFormulaValueIsLocalOrRemote'];
 | 
			
		||||
export type RoleCondFormulaValueUserSettingBooleanSchema = components['schemas']['RoleCondFormulaValueUserSettingBooleanSchema'];
 | 
			
		||||
export type RoleCondFormulaValueAssignedRole = components['schemas']['RoleCondFormulaValueAssignedRole'];
 | 
			
		||||
export type RoleCondFormulaValueCreated = components['schemas']['RoleCondFormulaValueCreated'];
 | 
			
		||||
export type RoleCondFormulaFollowersOrFollowingOrNotes = components['schemas']['RoleCondFormulaFollowersOrFollowingOrNotes'];
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -4603,6 +4603,7 @@ export type components = {
 | 
			
		|||
      isPublic: boolean;
 | 
			
		||||
      favoritedCount: number;
 | 
			
		||||
      isFavorited?: boolean;
 | 
			
		||||
      notesCount?: number;
 | 
			
		||||
    };
 | 
			
		||||
    FederationInstance: {
 | 
			
		||||
      /** Format: id */
 | 
			
		||||
| 
						 | 
				
			
			@ -4729,6 +4730,11 @@ export type components = {
 | 
			
		|||
      /** @enum {string} */
 | 
			
		||||
      type: 'isLocal' | 'isRemote';
 | 
			
		||||
    };
 | 
			
		||||
    RoleCondFormulaValueUserSettingBooleanSchema: {
 | 
			
		||||
      id: string;
 | 
			
		||||
      /** @enum {string} */
 | 
			
		||||
      type: 'isSuspended' | 'isLocked' | 'isBot' | 'isCat' | 'isExplorable';
 | 
			
		||||
    };
 | 
			
		||||
    RoleCondFormulaValueAssignedRole: {
 | 
			
		||||
      id: string;
 | 
			
		||||
      /** @enum {string} */
 | 
			
		||||
| 
						 | 
				
			
			@ -4751,7 +4757,7 @@ export type components = {
 | 
			
		|||
      type: 'followersLessThanOrEq' | 'followersMoreThanOrEq' | 'followingLessThanOrEq' | 'followingMoreThanOrEq' | 'notesLessThanOrEq' | 'notesMoreThanOrEq';
 | 
			
		||||
      value: number;
 | 
			
		||||
    };
 | 
			
		||||
    RoleCondFormulaValue: components['schemas']['RoleCondFormulaLogics'] | components['schemas']['RoleCondFormulaValueNot'] | components['schemas']['RoleCondFormulaValueIsLocalOrRemote'] | components['schemas']['RoleCondFormulaValueAssignedRole'] | components['schemas']['RoleCondFormulaValueCreated'] | components['schemas']['RoleCondFormulaFollowersOrFollowingOrNotes'];
 | 
			
		||||
    RoleCondFormulaValue: components['schemas']['RoleCondFormulaLogics'] | components['schemas']['RoleCondFormulaValueNot'] | components['schemas']['RoleCondFormulaValueIsLocalOrRemote'] | components['schemas']['RoleCondFormulaValueUserSettingBooleanSchema'] | components['schemas']['RoleCondFormulaValueAssignedRole'] | components['schemas']['RoleCondFormulaValueCreated'] | components['schemas']['RoleCondFormulaFollowersOrFollowingOrNotes'];
 | 
			
		||||
    RoleLite: {
 | 
			
		||||
      /**
 | 
			
		||||
       * Format: id
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										15
									
								
								pnpm-lock.yaml
									
										
									
										generated
									
									
									
								
							
							
						
						
									
										15
									
								
								pnpm-lock.yaml
									
										
									
										generated
									
									
									
								
							| 
						 | 
				
			
			@ -715,8 +715,8 @@ importers:
 | 
			
		|||
        specifier: 5.1.0
 | 
			
		||||
        version: 5.1.0(rollup@4.12.0)
 | 
			
		||||
      '@syuilo/aiscript':
 | 
			
		||||
        specifier: 0.17.0
 | 
			
		||||
        version: 0.17.0
 | 
			
		||||
        specifier: 0.18.0
 | 
			
		||||
        version: 0.18.0
 | 
			
		||||
      '@transfem-org/sfm-js':
 | 
			
		||||
        specifier: 0.24.4
 | 
			
		||||
        version: 0.24.4
 | 
			
		||||
| 
						 | 
				
			
			@ -6968,7 +6968,7 @@ packages:
 | 
			
		|||
      ts-dedent: 2.2.0
 | 
			
		||||
      type-fest: 2.19.0
 | 
			
		||||
      vue: 3.4.21(typescript@5.3.3)
 | 
			
		||||
      vue-component-type-helpers: 2.0.12
 | 
			
		||||
      vue-component-type-helpers: 2.0.14
 | 
			
		||||
    transitivePeerDependencies:
 | 
			
		||||
      - encoding
 | 
			
		||||
      - supports-color
 | 
			
		||||
| 
						 | 
				
			
			@ -7234,8 +7234,8 @@ packages:
 | 
			
		|||
    dev: false
 | 
			
		||||
    optional: true
 | 
			
		||||
 | 
			
		||||
  /@syuilo/aiscript@0.17.0:
 | 
			
		||||
    resolution: {integrity: sha512-3JtQ1rWJHMxQ3153zLCXMUOwrOgjPPYGBl0dPHhR0ohm4tn7okMQRugxMCT0t3YxByemb9FfiM6TUjd0tEGxdA==}
 | 
			
		||||
  /@syuilo/aiscript@0.18.0:
 | 
			
		||||
    resolution: {integrity: sha512-/iY9Vv4LLjtW/KUzId1QwXC4BlpIEPCMcoT7dyRhYdyxtwhS3Hx4b/4j1HYP+n3Pq9XKyW5zvkY72/+DNu4g6Q==}
 | 
			
		||||
    dependencies:
 | 
			
		||||
      seedrandom: 3.0.5
 | 
			
		||||
      stringz: 2.1.0
 | 
			
		||||
| 
						 | 
				
			
			@ -17655,6 +17655,7 @@ packages:
 | 
			
		|||
    hasBin: true
 | 
			
		||||
    dependencies:
 | 
			
		||||
      glob: 10.3.10
 | 
			
		||||
    dev: true
 | 
			
		||||
 | 
			
		||||
  /rollup@4.12.0:
 | 
			
		||||
    resolution: {integrity: sha512-wz66wn4t1OHIJw3+XU7mJJQV/2NAfw5OAk6G6Hoo3zcvz/XOfQ52Vgi+AN4Uxoxi0KBBwk2g8zPrTDA4btSB/Q==}
 | 
			
		||||
| 
						 | 
				
			
			@ -19788,8 +19789,8 @@ packages:
 | 
			
		|||
    resolution: {integrity: sha512-6bnLkn8O0JJyiFSIF0EfCogzeqNXpnjJ0vW/SZzNHfe6sPx30lTtTXlE5TFs2qhJlAtDFybStVNpL73cPe3OMQ==}
 | 
			
		||||
    dev: true
 | 
			
		||||
 | 
			
		||||
  /vue-component-type-helpers@2.0.12:
 | 
			
		||||
    resolution: {integrity: sha512-iVJugClQdu3ZyF0N4CF3Egi+gWYfnxlIPPGtFXZG29rF3kQIuziP+k7rVGCCHiibIOQ1SlspKjrh+LRYzMpwTA==}
 | 
			
		||||
  /vue-component-type-helpers@2.0.14:
 | 
			
		||||
    resolution: {integrity: sha512-DInfgOyXlMyliyqAAD9frK28tTfch0+tMi4qoWJcZlRxUf+NFAtraJBnAsKLep+FOyLMiajkhfyEb3xLK08i7w==}
 | 
			
		||||
    dev: true
 | 
			
		||||
 | 
			
		||||
  /vue-demi@0.14.7(vue@3.4.21):
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue