Merge remote-tracking branch 'misskey/develop' into future-2024-03-14
This commit is contained in:
		
						commit
						9478fc0095
					
				
					 57 changed files with 1082 additions and 81 deletions
				
			
		| 
						 | 
					@ -38,7 +38,7 @@
 | 
				
			||||||
# Option 3: If neither of the above applies to you.
 | 
					# Option 3: If neither of the above applies to you.
 | 
				
			||||||
#           (In this case, the source code should be published
 | 
					#           (In this case, the source code should be published
 | 
				
			||||||
#           on the Misskey interface.  IT IS NOT ENOUGH TO
 | 
					#           on the Misskey interface.  IT IS NOT ENOUGH TO
 | 
				
			||||||
#           DISCLOSE THE SOURCE CODE WEHN A USER REQUESTS IT BY
 | 
					#           DISCLOSE THE SOURCE CODE WHEN A USER REQUESTS IT BY
 | 
				
			||||||
#           E-MAIL OR OTHER MEANS.  If you are not satisfied
 | 
					#           E-MAIL OR OTHER MEANS.  If you are not satisfied
 | 
				
			||||||
#           with this, it is recommended that you read the
 | 
					#           with this, it is recommended that you read the
 | 
				
			||||||
#           license again carefully.  Anyway, enabling this
 | 
					#           license again carefully.  Anyway, enabling this
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -19,7 +19,6 @@
 | 
				
			||||||
				"editorconfig.editorconfig",
 | 
									"editorconfig.editorconfig",
 | 
				
			||||||
				"dbaeumer.vscode-eslint",
 | 
									"dbaeumer.vscode-eslint",
 | 
				
			||||||
				"Vue.volar",
 | 
									"Vue.volar",
 | 
				
			||||||
				"Vue.vscode-typescript-vue-plugin",
 | 
					 | 
				
			||||||
				"Orta.vscode-jest",
 | 
									"Orta.vscode-jest",
 | 
				
			||||||
				"dbaeumer.vscode-eslint",
 | 
									"dbaeumer.vscode-eslint",
 | 
				
			||||||
				"mrmlnc.vscode-json5"
 | 
									"mrmlnc.vscode-json5"
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
							
								
								
									
										75
									
								
								.github/workflows/check-spdx-license-id.yml
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										75
									
								
								.github/workflows/check-spdx-license-id.yml
									
										
									
									
										vendored
									
									
										Normal file
									
								
							| 
						 | 
					@ -0,0 +1,75 @@
 | 
				
			||||||
 | 
					name: Check SPDX-License-Identifier
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					on:
 | 
				
			||||||
 | 
					  push:
 | 
				
			||||||
 | 
					    branches:
 | 
				
			||||||
 | 
					      - master
 | 
				
			||||||
 | 
					      - develop
 | 
				
			||||||
 | 
					  pull_request:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					jobs:
 | 
				
			||||||
 | 
					  check-spdx-license-id:
 | 
				
			||||||
 | 
					    runs-on: ubuntu-latest
 | 
				
			||||||
 | 
					    steps:
 | 
				
			||||||
 | 
					      - name: Checkout
 | 
				
			||||||
 | 
					        uses: actions/checkout@v4.1.1
 | 
				
			||||||
 | 
					      - name: Check
 | 
				
			||||||
 | 
					        run: |
 | 
				
			||||||
 | 
					          counter=0
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					          search() {
 | 
				
			||||||
 | 
					            local directory="$1"
 | 
				
			||||||
 | 
					            find "$directory" -type f \
 | 
				
			||||||
 | 
					              '(' \
 | 
				
			||||||
 | 
					                -name "*.cjs" -and -not -name '*.config.cjs' -o \
 | 
				
			||||||
 | 
					                -name "*.html" -o \
 | 
				
			||||||
 | 
					                -name "*.js" -and -not -name '*.config.js' -o \
 | 
				
			||||||
 | 
					                -name "*.mjs" -and -not -name '*.config.mjs' -o \
 | 
				
			||||||
 | 
					                -name "*.scss" -o \
 | 
				
			||||||
 | 
					                -name "*.ts" -and -not -name '*.config.ts' -o \
 | 
				
			||||||
 | 
					                -name "*.vue" \
 | 
				
			||||||
 | 
					              ')' -and \
 | 
				
			||||||
 | 
					              -not -name '*eslint*'
 | 
				
			||||||
 | 
					          }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					          check() {
 | 
				
			||||||
 | 
					            local file="$1"
 | 
				
			||||||
 | 
					            if ! (
 | 
				
			||||||
 | 
					              grep -q "SPDX-FileCopyrightText: syuilo and misskey-project" "$file" ||
 | 
				
			||||||
 | 
					              grep -q "SPDX-License-Identifier: AGPL-3.0-only" "$file"
 | 
				
			||||||
 | 
					            ); then
 | 
				
			||||||
 | 
					              echo "Missing: $file"
 | 
				
			||||||
 | 
					              ((counter++))
 | 
				
			||||||
 | 
					            fi
 | 
				
			||||||
 | 
					          }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					          directories=(
 | 
				
			||||||
 | 
					            "cypress/e2e"
 | 
				
			||||||
 | 
					            "packages/backend/migration"
 | 
				
			||||||
 | 
					            "packages/backend/src"
 | 
				
			||||||
 | 
					            "packages/backend/test"
 | 
				
			||||||
 | 
					            "packages/frontend/.storybook"
 | 
				
			||||||
 | 
					            "packages/frontend/@types"
 | 
				
			||||||
 | 
					            "packages/frontend/lib"
 | 
				
			||||||
 | 
					            "packages/frontend/public"
 | 
				
			||||||
 | 
					            "packages/frontend/src"
 | 
				
			||||||
 | 
					            "packages/frontend/test"
 | 
				
			||||||
 | 
					            "packages/misskey-bubble-game/src"
 | 
				
			||||||
 | 
					            "packages/misskey-reversi/src"
 | 
				
			||||||
 | 
					            "packages/sw/src"
 | 
				
			||||||
 | 
					            "scripts"
 | 
				
			||||||
 | 
					          )
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					          for directory in "${directories[@]}"; do
 | 
				
			||||||
 | 
					            for file in $(search $directory); do
 | 
				
			||||||
 | 
					              check "$file"
 | 
				
			||||||
 | 
					            done
 | 
				
			||||||
 | 
					          done
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					          if [ $counter -gt 0 ]; then
 | 
				
			||||||
 | 
					            echo "SPDX-License-Identifier is missing in $counter files."
 | 
				
			||||||
 | 
					            exit 1
 | 
				
			||||||
 | 
					          else
 | 
				
			||||||
 | 
					            echo "SPDX-License-Identifier is certainly described in all target files!"
 | 
				
			||||||
 | 
					            exit 0
 | 
				
			||||||
 | 
					          fi
 | 
				
			||||||
							
								
								
									
										2
									
								
								.vscode/extensions.json
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.vscode/extensions.json
									
										
									
									
										vendored
									
									
								
							| 
						 | 
					@ -3,9 +3,7 @@
 | 
				
			||||||
		"editorconfig.editorconfig",
 | 
							"editorconfig.editorconfig",
 | 
				
			||||||
		"dbaeumer.vscode-eslint",
 | 
							"dbaeumer.vscode-eslint",
 | 
				
			||||||
		"Vue.volar",
 | 
							"Vue.volar",
 | 
				
			||||||
		"Vue.vscode-typescript-vue-plugin",
 | 
					 | 
				
			||||||
		"Orta.vscode-jest",
 | 
							"Orta.vscode-jest",
 | 
				
			||||||
		"dbaeumer.vscode-eslint",
 | 
					 | 
				
			||||||
		"mrmlnc.vscode-json5"
 | 
							"mrmlnc.vscode-json5"
 | 
				
			||||||
	]
 | 
						]
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
							
								
								
									
										2
									
								
								.vscode/settings.json
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.vscode/settings.json
									
										
									
									
										vendored
									
									
								
							| 
						 | 
					@ -7,7 +7,7 @@
 | 
				
			||||||
		"*.test.ts": "typescript"
 | 
							"*.test.ts": "typescript"
 | 
				
			||||||
	},
 | 
						},
 | 
				
			||||||
	"jest.jestCommandLine": "pnpm run jest",
 | 
						"jest.jestCommandLine": "pnpm run jest",
 | 
				
			||||||
	"jest.autoRun": "off",
 | 
						"jest.runMode": "on-demand",
 | 
				
			||||||
	"editor.codeActionsOnSave": {
 | 
						"editor.codeActionsOnSave": {
 | 
				
			||||||
		"source.fixAll": "explicit"
 | 
							"source.fixAll": "explicit"
 | 
				
			||||||
	},
 | 
						},
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -8,11 +8,14 @@
 | 
				
			||||||
- Enhance: 広告がMisskeyと同一ドメインの場合はRouterで遷移するように
 | 
					- Enhance: 広告がMisskeyと同一ドメインの場合はRouterで遷移するように
 | 
				
			||||||
- Enhance: リアクション・いいねの総数を表示するように
 | 
					- Enhance: リアクション・いいねの総数を表示するように
 | 
				
			||||||
- Enhance: リアクション受け入れが「いいねのみ」の場合はリアクション絵文字一覧を表示しないように
 | 
					- Enhance: リアクション受け入れが「いいねのみ」の場合はリアクション絵文字一覧を表示しないように
 | 
				
			||||||
 | 
					- Enhance: 設定>プラグインのページからプラグインの簡易的なログやエラーを見られるように
 | 
				
			||||||
 | 
					  - 実装の都合により、プラグインは1つエラーを起こした時に即時停止するようになりました
 | 
				
			||||||
- Fix: 一部のページ内リンクが正しく動作しない問題を修正
 | 
					- Fix: 一部のページ内リンクが正しく動作しない問題を修正
 | 
				
			||||||
- Fix: 周年の実績が閏年を考慮しない問題を修正
 | 
					- Fix: 周年の実績が閏年を考慮しない問題を修正
 | 
				
			||||||
 | 
					- Fix: ローカルURLのプレビューポップアップが左上に表示される
 | 
				
			||||||
 | 
					
 | 
				
			||||||
### Server
 | 
					### Server
 | 
				
			||||||
-
 | 
					- Enhance: エンドポイント`antennas/update`の必須項目を`antennaId`のみに
 | 
				
			||||||
 | 
					
 | 
				
			||||||
## 2024.3.1
 | 
					## 2024.3.1
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
describe('Before setup instance', () => {
 | 
					describe('Before setup instance', () => {
 | 
				
			||||||
	beforeEach(() => {
 | 
						beforeEach(() => {
 | 
				
			||||||
		cy.resetState();
 | 
							cy.resetState();
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
describe('Router transition', () => {
 | 
					describe('Router transition', () => {
 | 
				
			||||||
	describe('Redirect', () => {
 | 
						describe('Redirect', () => {
 | 
				
			||||||
		// サーバの初期化。ルートのテストに関しては各describeごとに1度だけ実行で十分だと思う(使いまわした方が早い)
 | 
							// サーバの初期化。ルートのテストに関しては各describeごとに1度だけ実行で十分だと思う(使いまわした方が早い)
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/* flaky
 | 
					/* flaky
 | 
				
			||||||
describe('After user signed in', () => {
 | 
					describe('After user signed in', () => {
 | 
				
			||||||
	beforeEach(() => {
 | 
						beforeEach(() => {
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
							
								
								
									
										4
									
								
								locales/index.d.ts
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								locales/index.d.ts
									
										
									
									
										vendored
									
									
								
							| 
						 | 
					@ -7018,6 +7018,10 @@ export interface Locale extends ILocale {
 | 
				
			||||||
         * ソースを表示
 | 
					         * ソースを表示
 | 
				
			||||||
         */
 | 
					         */
 | 
				
			||||||
        "viewSource": string;
 | 
					        "viewSource": string;
 | 
				
			||||||
 | 
					        /**
 | 
				
			||||||
 | 
					         * ログを表示
 | 
				
			||||||
 | 
					         */
 | 
				
			||||||
 | 
					        "viewLog": string;
 | 
				
			||||||
    };
 | 
					    };
 | 
				
			||||||
    "_preferencesBackups": {
 | 
					    "_preferencesBackups": {
 | 
				
			||||||
        /**
 | 
					        /**
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1825,6 +1825,7 @@ _plugin:
 | 
				
			||||||
  installWarn: "信頼できないプラグインはインストールしないでください。"
 | 
					  installWarn: "信頼できないプラグインはインストールしないでください。"
 | 
				
			||||||
  manage: "プラグインの管理"
 | 
					  manage: "プラグインの管理"
 | 
				
			||||||
  viewSource: "ソースを表示"
 | 
					  viewSource: "ソースを表示"
 | 
				
			||||||
 | 
					  viewLog: "ログを表示"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
_preferencesBackups:
 | 
					_preferencesBackups:
 | 
				
			||||||
  list: "作成したバックアップ"
 | 
					  list: "作成したバックアップ"
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import { loadConfig } from './built/config.js'
 | 
					import { loadConfig } from './built/config.js'
 | 
				
			||||||
import { genOpenapiSpec } from './built/server/api/openapi/gen-spec.js'
 | 
					import { genOpenapiSpec } from './built/server/api/openapi/gen-spec.js'
 | 
				
			||||||
import { writeFileSync } from "node:fs";
 | 
					import { writeFileSync } from "node:fs";
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export class UserBlacklistAnntena1689325027964 {
 | 
					export class UserBlacklistAnntena1689325027964 {
 | 
				
			||||||
    name = 'UserBlacklistAnntena1689325027964'
 | 
					    name = 'UserBlacklistAnntena1689325027964'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export class FixRenoteMuting1690417561185 {
 | 
					export class FixRenoteMuting1690417561185 {
 | 
				
			||||||
    name = 'FixRenoteMuting1690417561185'
 | 
					    name = 'FixRenoteMuting1690417561185'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export class ChangeCacheRemoteFilesDefault1690417561186 {
 | 
					export class ChangeCacheRemoteFilesDefault1690417561186 {
 | 
				
			||||||
    name = 'ChangeCacheRemoteFilesDefault1690417561186'
 | 
					    name = 'ChangeCacheRemoteFilesDefault1690417561186'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export class Fix1690417561187 {
 | 
					export class Fix1690417561187 {
 | 
				
			||||||
    name = 'Fix1690417561187'
 | 
					    name = 'Fix1690417561187'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export class User2faBackupCodes1690569881926 {
 | 
					export class User2faBackupCodes1690569881926 {
 | 
				
			||||||
	name = 'User2faBackupCodes1690569881926'
 | 
						name = 'User2faBackupCodes1690569881926'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export class RefineAnnouncement1691649257651 {
 | 
					export class RefineAnnouncement1691649257651 {
 | 
				
			||||||
    name = 'RefineAnnouncement1691649257651'
 | 
					    name = 'RefineAnnouncement1691649257651'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export class RefineAnnouncement21691657412740 {
 | 
					export class RefineAnnouncement21691657412740 {
 | 
				
			||||||
    name = 'RefineAnnouncement21691657412740'
 | 
					    name = 'RefineAnnouncement21691657412740'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export class VerifiedLinks1695260774117 {
 | 
					export class VerifiedLinks1695260774117 {
 | 
				
			||||||
    name = 'VerifiedLinks1695260774117'
 | 
					    name = 'VerifiedLinks1695260774117'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export class FollowingNotify1695288787870 {
 | 
					export class FollowingNotify1695288787870 {
 | 
				
			||||||
    name = 'FollowingNotify1695288787870'
 | 
					    name = 'FollowingNotify1695288787870'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export class ShortName1695440131671 {
 | 
					export class ShortName1695440131671 {
 | 
				
			||||||
    name = 'ShortName1695440131671'
 | 
					    name = 'ShortName1695440131671'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export class MutingNotificationTypes1695605508898 {
 | 
					export class MutingNotificationTypes1695605508898 {
 | 
				
			||||||
    name = 'MutingNotificationTypes1695605508898'
 | 
					    name = 'MutingNotificationTypes1695605508898'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export class UserListMembership1696323464251 {
 | 
					export class UserListMembership1696323464251 {
 | 
				
			||||||
    name = 'UserListMembership1696323464251'
 | 
					    name = 'UserListMembership1696323464251'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export class Hibernation1696331570827 {
 | 
					export class Hibernation1696331570827 {
 | 
				
			||||||
    name = 'Hibernation1696331570827'
 | 
					    name = 'Hibernation1696331570827'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export class Clean1696332072038 {
 | 
					export class Clean1696332072038 {
 | 
				
			||||||
    name = 'Clean1696332072038'
 | 
					    name = 'Clean1696332072038'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export class HardMute1700383825690 {
 | 
					export class HardMute1700383825690 {
 | 
				
			||||||
    name = 'HardMute1700383825690'
 | 
					    name = 'HardMute1700383825690'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import { Inject, Injectable, OnModuleInit } from '@nestjs/common';
 | 
					import { Inject, Injectable, OnModuleInit } from '@nestjs/common';
 | 
				
			||||||
import Redis from 'ioredis';
 | 
					import Redis from 'ioredis';
 | 
				
			||||||
import { DI } from '@/di-symbols.js';
 | 
					import { DI } from '@/di-symbols.js';
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -134,7 +134,7 @@ export class ApNoteService {
 | 
				
			||||||
				value,
 | 
									value,
 | 
				
			||||||
				object,
 | 
									object,
 | 
				
			||||||
			});
 | 
								});
 | 
				
			||||||
			throw new Error('invalid note');
 | 
								throw err;
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		const note = object as IPost;
 | 
							const note = object as IPost;
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -7,6 +7,7 @@ import { Inject, Injectable } from '@nestjs/common';
 | 
				
			||||||
import * as Redis from 'ioredis';
 | 
					import * as Redis from 'ioredis';
 | 
				
			||||||
import _Ajv from 'ajv';
 | 
					import _Ajv from 'ajv';
 | 
				
			||||||
import { ModuleRef } from '@nestjs/core';
 | 
					import { ModuleRef } from '@nestjs/core';
 | 
				
			||||||
 | 
					import { In } from 'typeorm';
 | 
				
			||||||
import { DI } from '@/di-symbols.js';
 | 
					import { DI } from '@/di-symbols.js';
 | 
				
			||||||
import type { Config } from '@/config.js';
 | 
					import type { Config } from '@/config.js';
 | 
				
			||||||
import type { Packed } from '@/misc/json-schema.js';
 | 
					import type { Packed } from '@/misc/json-schema.js';
 | 
				
			||||||
| 
						 | 
					@ -14,9 +15,31 @@ import type { Promiseable } from '@/misc/prelude/await-all.js';
 | 
				
			||||||
import { awaitAll } from '@/misc/prelude/await-all.js';
 | 
					import { awaitAll } from '@/misc/prelude/await-all.js';
 | 
				
			||||||
import { USER_ACTIVE_THRESHOLD, USER_ONLINE_THRESHOLD } from '@/const.js';
 | 
					import { USER_ACTIVE_THRESHOLD, USER_ONLINE_THRESHOLD } from '@/const.js';
 | 
				
			||||||
import type { MiLocalUser, MiPartialLocalUser, MiPartialRemoteUser, MiRemoteUser, MiUser } from '@/models/User.js';
 | 
					import type { MiLocalUser, MiPartialLocalUser, MiPartialRemoteUser, MiRemoteUser, MiUser } from '@/models/User.js';
 | 
				
			||||||
import { birthdaySchema, listenbrainzSchema, descriptionSchema, localUsernameSchema, locationSchema, nameSchema, passwordSchema } from '@/models/User.js';
 | 
					import {
 | 
				
			||||||
import { MiNotification } from '@/models/Notification.js';
 | 
						birthdaySchema,
 | 
				
			||||||
import type { UsersRepository, UserSecurityKeysRepository, FollowingsRepository, FollowRequestsRepository, BlockingsRepository, MutingsRepository, DriveFilesRepository, NoteUnreadsRepository, UserNotePiningsRepository, UserProfilesRepository, AnnouncementReadsRepository, AnnouncementsRepository, MiUserProfile, RenoteMutingsRepository, UserMemoRepository } from '@/models/_.js';
 | 
						descriptionSchema,
 | 
				
			||||||
 | 
					  listenbrainzSchema,
 | 
				
			||||||
 | 
					  localUsernameSchema,
 | 
				
			||||||
 | 
						locationSchema,
 | 
				
			||||||
 | 
						nameSchema,
 | 
				
			||||||
 | 
						passwordSchema,
 | 
				
			||||||
 | 
					} from '@/models/User.js';
 | 
				
			||||||
 | 
					import type {
 | 
				
			||||||
 | 
						BlockingsRepository,
 | 
				
			||||||
 | 
						FollowingsRepository,
 | 
				
			||||||
 | 
						FollowRequestsRepository,
 | 
				
			||||||
 | 
						MiFollowing,
 | 
				
			||||||
 | 
						MiUserNotePining,
 | 
				
			||||||
 | 
						MiUserProfile,
 | 
				
			||||||
 | 
						MutingsRepository,
 | 
				
			||||||
 | 
						NoteUnreadsRepository,
 | 
				
			||||||
 | 
						RenoteMutingsRepository,
 | 
				
			||||||
 | 
						UserMemoRepository,
 | 
				
			||||||
 | 
						UserNotePiningsRepository,
 | 
				
			||||||
 | 
						UserProfilesRepository,
 | 
				
			||||||
 | 
						UserSecurityKeysRepository,
 | 
				
			||||||
 | 
						UsersRepository,
 | 
				
			||||||
 | 
					} from '@/models/_.js';
 | 
				
			||||||
import { bindThis } from '@/decorators.js';
 | 
					import { bindThis } from '@/decorators.js';
 | 
				
			||||||
import { RoleService } from '@/core/RoleService.js';
 | 
					import { RoleService } from '@/core/RoleService.js';
 | 
				
			||||||
import { ApPersonService } from '@/core/activitypub/models/ApPersonService.js';
 | 
					import { ApPersonService } from '@/core/activitypub/models/ApPersonService.js';
 | 
				
			||||||
| 
						 | 
					@ -46,11 +69,23 @@ function isRemoteUser(user: MiUser | { host: MiUser['host'] }): boolean {
 | 
				
			||||||
	return !isLocalUser(user);
 | 
						return !isLocalUser(user);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export type UserRelation = {
 | 
				
			||||||
 | 
						id: MiUser['id']
 | 
				
			||||||
 | 
						following: MiFollowing | null,
 | 
				
			||||||
 | 
						isFollowing: boolean
 | 
				
			||||||
 | 
						isFollowed: boolean
 | 
				
			||||||
 | 
						hasPendingFollowRequestFromYou: boolean
 | 
				
			||||||
 | 
						hasPendingFollowRequestToYou: boolean
 | 
				
			||||||
 | 
						isBlocking: boolean
 | 
				
			||||||
 | 
						isBlocked: boolean
 | 
				
			||||||
 | 
						isMuted: boolean
 | 
				
			||||||
 | 
						isRenoteMuted: boolean
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@Injectable()
 | 
					@Injectable()
 | 
				
			||||||
export class UserEntityService implements OnModuleInit {
 | 
					export class UserEntityService implements OnModuleInit {
 | 
				
			||||||
	private apPersonService: ApPersonService;
 | 
						private apPersonService: ApPersonService;
 | 
				
			||||||
	private noteEntityService: NoteEntityService;
 | 
						private noteEntityService: NoteEntityService;
 | 
				
			||||||
	private driveFileEntityService: DriveFileEntityService;
 | 
					 | 
				
			||||||
	private pageEntityService: PageEntityService;
 | 
						private pageEntityService: PageEntityService;
 | 
				
			||||||
	private customEmojiService: CustomEmojiService;
 | 
						private customEmojiService: CustomEmojiService;
 | 
				
			||||||
	private announcementService: AnnouncementService;
 | 
						private announcementService: AnnouncementService;
 | 
				
			||||||
| 
						 | 
					@ -89,9 +124,6 @@ export class UserEntityService implements OnModuleInit {
 | 
				
			||||||
		@Inject(DI.renoteMutingsRepository)
 | 
							@Inject(DI.renoteMutingsRepository)
 | 
				
			||||||
		private renoteMutingsRepository: RenoteMutingsRepository,
 | 
							private renoteMutingsRepository: RenoteMutingsRepository,
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		@Inject(DI.driveFilesRepository)
 | 
					 | 
				
			||||||
		private driveFilesRepository: DriveFilesRepository,
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		@Inject(DI.noteUnreadsRepository)
 | 
							@Inject(DI.noteUnreadsRepository)
 | 
				
			||||||
		private noteUnreadsRepository: NoteUnreadsRepository,
 | 
							private noteUnreadsRepository: NoteUnreadsRepository,
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -101,12 +133,6 @@ export class UserEntityService implements OnModuleInit {
 | 
				
			||||||
		@Inject(DI.userProfilesRepository)
 | 
							@Inject(DI.userProfilesRepository)
 | 
				
			||||||
		private userProfilesRepository: UserProfilesRepository,
 | 
							private userProfilesRepository: UserProfilesRepository,
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		@Inject(DI.announcementReadsRepository)
 | 
					 | 
				
			||||||
		private announcementReadsRepository: AnnouncementReadsRepository,
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		@Inject(DI.announcementsRepository)
 | 
					 | 
				
			||||||
		private announcementsRepository: AnnouncementsRepository,
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		@Inject(DI.userMemosRepository)
 | 
							@Inject(DI.userMemosRepository)
 | 
				
			||||||
		private userMemosRepository: UserMemoRepository,
 | 
							private userMemosRepository: UserMemoRepository,
 | 
				
			||||||
	) {
 | 
						) {
 | 
				
			||||||
| 
						 | 
					@ -115,7 +141,6 @@ export class UserEntityService implements OnModuleInit {
 | 
				
			||||||
	onModuleInit() {
 | 
						onModuleInit() {
 | 
				
			||||||
		this.apPersonService = this.moduleRef.get('ApPersonService');
 | 
							this.apPersonService = this.moduleRef.get('ApPersonService');
 | 
				
			||||||
		this.noteEntityService = this.moduleRef.get('NoteEntityService');
 | 
							this.noteEntityService = this.moduleRef.get('NoteEntityService');
 | 
				
			||||||
		this.driveFileEntityService = this.moduleRef.get('DriveFileEntityService');
 | 
					 | 
				
			||||||
		this.pageEntityService = this.moduleRef.get('PageEntityService');
 | 
							this.pageEntityService = this.moduleRef.get('PageEntityService');
 | 
				
			||||||
		this.customEmojiService = this.moduleRef.get('CustomEmojiService');
 | 
							this.customEmojiService = this.moduleRef.get('CustomEmojiService');
 | 
				
			||||||
		this.announcementService = this.moduleRef.get('AnnouncementService');
 | 
							this.announcementService = this.moduleRef.get('AnnouncementService');
 | 
				
			||||||
| 
						 | 
					@ -139,7 +164,7 @@ export class UserEntityService implements OnModuleInit {
 | 
				
			||||||
	public isRemoteUser = isRemoteUser;
 | 
						public isRemoteUser = isRemoteUser;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	@bindThis
 | 
						@bindThis
 | 
				
			||||||
	public async getRelation(me: MiUser['id'], target: MiUser['id']) {
 | 
						public async getRelation(me: MiUser['id'], target: MiUser['id']): Promise<UserRelation> {
 | 
				
			||||||
		const [
 | 
							const [
 | 
				
			||||||
			following,
 | 
								following,
 | 
				
			||||||
			isFollowed,
 | 
								isFollowed,
 | 
				
			||||||
| 
						 | 
					@ -212,6 +237,59 @@ export class UserEntityService implements OnModuleInit {
 | 
				
			||||||
		};
 | 
							};
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						@bindThis
 | 
				
			||||||
 | 
						public async getRelations(me: MiUser['id'], targets: MiUser['id'][]): Promise<Map<MiUser['id'], UserRelation>> {
 | 
				
			||||||
 | 
							const [
 | 
				
			||||||
 | 
								followers,
 | 
				
			||||||
 | 
								followees,
 | 
				
			||||||
 | 
								followersRequests,
 | 
				
			||||||
 | 
								followeesRequests,
 | 
				
			||||||
 | 
								blockers,
 | 
				
			||||||
 | 
								blockees,
 | 
				
			||||||
 | 
								muters,
 | 
				
			||||||
 | 
								renoteMuters,
 | 
				
			||||||
 | 
							] = await Promise.all([
 | 
				
			||||||
 | 
								this.followingsRepository.findBy({ followerId: me })
 | 
				
			||||||
 | 
									.then(f => new Map(f.map(it => [it.followeeId, it]))),
 | 
				
			||||||
 | 
								this.followingsRepository.findBy({ followeeId: me })
 | 
				
			||||||
 | 
									.then(it => it.map(it => it.followerId)),
 | 
				
			||||||
 | 
								this.followRequestsRepository.findBy({ followerId: me })
 | 
				
			||||||
 | 
									.then(it => it.map(it => it.followeeId)),
 | 
				
			||||||
 | 
								this.followRequestsRepository.findBy({ followeeId: me })
 | 
				
			||||||
 | 
									.then(it => it.map(it => it.followerId)),
 | 
				
			||||||
 | 
								this.blockingsRepository.findBy({ blockerId: me })
 | 
				
			||||||
 | 
									.then(it => it.map(it => it.blockeeId)),
 | 
				
			||||||
 | 
								this.blockingsRepository.findBy({ blockeeId: me })
 | 
				
			||||||
 | 
									.then(it => it.map(it => it.blockerId)),
 | 
				
			||||||
 | 
								this.mutingsRepository.findBy({ muterId: me })
 | 
				
			||||||
 | 
									.then(it => it.map(it => it.muteeId)),
 | 
				
			||||||
 | 
								this.renoteMutingsRepository.findBy({ muterId: me })
 | 
				
			||||||
 | 
									.then(it => it.map(it => it.muteeId)),
 | 
				
			||||||
 | 
							]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							return new Map(
 | 
				
			||||||
 | 
								targets.map(target => {
 | 
				
			||||||
 | 
									const following = followers.get(target) ?? null;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
									return [
 | 
				
			||||||
 | 
										target,
 | 
				
			||||||
 | 
										{
 | 
				
			||||||
 | 
											id: target,
 | 
				
			||||||
 | 
											following: following,
 | 
				
			||||||
 | 
											isFollowing: following != null,
 | 
				
			||||||
 | 
											isFollowed: followees.includes(target),
 | 
				
			||||||
 | 
											hasPendingFollowRequestFromYou: followersRequests.includes(target),
 | 
				
			||||||
 | 
											hasPendingFollowRequestToYou: followeesRequests.includes(target),
 | 
				
			||||||
 | 
											isBlocking: blockers.includes(target),
 | 
				
			||||||
 | 
											isBlocked: blockees.includes(target),
 | 
				
			||||||
 | 
											isMuted: muters.includes(target),
 | 
				
			||||||
 | 
											isRenoteMuted: renoteMuters.includes(target),
 | 
				
			||||||
 | 
										},
 | 
				
			||||||
 | 
									];
 | 
				
			||||||
 | 
								}),
 | 
				
			||||||
 | 
							);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	@bindThis
 | 
						@bindThis
 | 
				
			||||||
	public async getHasUnreadAntenna(userId: MiUser['id']): Promise<boolean> {
 | 
						public async getHasUnreadAntenna(userId: MiUser['id']): Promise<boolean> {
 | 
				
			||||||
		/*
 | 
							/*
 | 
				
			||||||
| 
						 | 
					@ -304,6 +382,9 @@ export class UserEntityService implements OnModuleInit {
 | 
				
			||||||
			schema?: S,
 | 
								schema?: S,
 | 
				
			||||||
			includeSecrets?: boolean,
 | 
								includeSecrets?: boolean,
 | 
				
			||||||
			userProfile?: MiUserProfile,
 | 
								userProfile?: MiUserProfile,
 | 
				
			||||||
 | 
								userRelations?: Map<MiUser['id'], UserRelation>,
 | 
				
			||||||
 | 
								userMemos?: Map<MiUser['id'], string | null>,
 | 
				
			||||||
 | 
								pinNotes?: Map<MiUser['id'], MiUserNotePining[]>,
 | 
				
			||||||
		},
 | 
							},
 | 
				
			||||||
	): Promise<Packed<S>> {
 | 
						): Promise<Packed<S>> {
 | 
				
			||||||
		const opts = Object.assign({
 | 
							const opts = Object.assign({
 | 
				
			||||||
| 
						 | 
					@ -344,13 +425,41 @@ export class UserEntityService implements OnModuleInit {
 | 
				
			||||||
		const isMe = meId === user.id;
 | 
							const isMe = meId === user.id;
 | 
				
			||||||
		const iAmModerator = me ? await this.roleService.isModerator(me as MiUser) : false;
 | 
							const iAmModerator = me ? await this.roleService.isModerator(me as MiUser) : false;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		const relation = meId && !isMe && isDetailed ? await this.getRelation(meId, user.id) : null;
 | 
							const profile = isDetailed
 | 
				
			||||||
		const pins = isDetailed ? await this.userNotePiningsRepository.createQueryBuilder('pin')
 | 
								? (opts.userProfile ?? await this.userProfilesRepository.findOneByOrFail({ userId: user.id }))
 | 
				
			||||||
 | 
								: null;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							let relation: UserRelation | null = null;
 | 
				
			||||||
 | 
							if (meId && !isMe && isDetailed) {
 | 
				
			||||||
 | 
								if (opts.userRelations) {
 | 
				
			||||||
 | 
									relation = opts.userRelations.get(user.id) ?? null;
 | 
				
			||||||
 | 
								} else {
 | 
				
			||||||
 | 
									relation = await this.getRelation(meId, user.id);
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							let memo: string | null = null;
 | 
				
			||||||
 | 
							if (isDetailed && meId) {
 | 
				
			||||||
 | 
								if (opts.userMemos) {
 | 
				
			||||||
 | 
									memo = opts.userMemos.get(user.id) ?? null;
 | 
				
			||||||
 | 
								} else {
 | 
				
			||||||
 | 
									memo = await this.userMemosRepository.findOneBy({ userId: meId, targetUserId: user.id })
 | 
				
			||||||
 | 
										.then(row => row?.memo ?? null);
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							let pins: MiUserNotePining[] = [];
 | 
				
			||||||
 | 
							if (isDetailed) {
 | 
				
			||||||
 | 
								if (opts.pinNotes) {
 | 
				
			||||||
 | 
									pins = opts.pinNotes.get(user.id) ?? [];
 | 
				
			||||||
 | 
								} else {
 | 
				
			||||||
 | 
									pins = await this.userNotePiningsRepository.createQueryBuilder('pin')
 | 
				
			||||||
					.where('pin.userId = :userId', { userId: user.id })
 | 
										.where('pin.userId = :userId', { userId: user.id })
 | 
				
			||||||
					.innerJoinAndSelect('pin.note', 'note')
 | 
										.innerJoinAndSelect('pin.note', 'note')
 | 
				
			||||||
					.orderBy('pin.id', 'DESC')
 | 
										.orderBy('pin.id', 'DESC')
 | 
				
			||||||
			.getMany() : [];
 | 
										.getMany();
 | 
				
			||||||
		const profile = isDetailed ? (opts.userProfile ?? await this.userProfilesRepository.findOneByOrFail({ userId: user.id })) : null;
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		const mastoapi = !isDetailed ? opts.userProfile ?? await this.userProfilesRepository.findOneByOrFail({ userId: user.id }) : null;
 | 
							const mastoapi = !isDetailed ? opts.userProfile ?? await this.userProfilesRepository.findOneByOrFail({ userId: user.id }) : null;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -452,9 +561,7 @@ export class UserEntityService implements OnModuleInit {
 | 
				
			||||||
				twoFactorEnabled: profile!.twoFactorEnabled,
 | 
									twoFactorEnabled: profile!.twoFactorEnabled,
 | 
				
			||||||
				usePasswordLessLogin: profile!.usePasswordLessLogin,
 | 
									usePasswordLessLogin: profile!.usePasswordLessLogin,
 | 
				
			||||||
				securityKeys: profile!.twoFactorEnabled
 | 
									securityKeys: profile!.twoFactorEnabled
 | 
				
			||||||
					? this.userSecurityKeysRepository.countBy({
 | 
										? this.userSecurityKeysRepository.countBy({ userId: user.id }).then(result => result >= 1)
 | 
				
			||||||
						userId: user.id,
 | 
					 | 
				
			||||||
					}).then(result => result >= 1)
 | 
					 | 
				
			||||||
					: false,
 | 
										: false,
 | 
				
			||||||
				roles: this.roleService.getUserRoles(user.id).then(roles => roles.filter(role => role.isPublic).sort((a, b) => b.displayOrder - a.displayOrder).map(role => ({
 | 
									roles: this.roleService.getUserRoles(user.id).then(roles => roles.filter(role => role.isPublic).sort((a, b) => b.displayOrder - a.displayOrder).map(role => ({
 | 
				
			||||||
					id: role.id,
 | 
										id: role.id,
 | 
				
			||||||
| 
						 | 
					@ -466,10 +573,7 @@ export class UserEntityService implements OnModuleInit {
 | 
				
			||||||
					isAdministrator: role.isAdministrator,
 | 
										isAdministrator: role.isAdministrator,
 | 
				
			||||||
					displayOrder: role.displayOrder,
 | 
										displayOrder: role.displayOrder,
 | 
				
			||||||
				}))),
 | 
									}))),
 | 
				
			||||||
				memo: meId == null ? null : await this.userMemosRepository.findOneBy({
 | 
									memo: memo,
 | 
				
			||||||
					userId: meId,
 | 
					 | 
				
			||||||
					targetUserId: user.id,
 | 
					 | 
				
			||||||
				}).then(row => row?.memo ?? null),
 | 
					 | 
				
			||||||
				moderationNote: iAmModerator ? (profile!.moderationNote ?? '') : undefined,
 | 
									moderationNote: iAmModerator ? (profile!.moderationNote ?? '') : undefined,
 | 
				
			||||||
			} : {}),
 | 
								} : {}),
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -552,7 +656,7 @@ export class UserEntityService implements OnModuleInit {
 | 
				
			||||||
		return await awaitAll(packed);
 | 
							return await awaitAll(packed);
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	public packMany<S extends 'MeDetailed' | 'UserDetailedNotMe' | 'UserDetailed' | 'UserLite' = 'UserLite'>(
 | 
						public async packMany<S extends 'MeDetailed' | 'UserDetailedNotMe' | 'UserDetailed' | 'UserLite' = 'UserLite'>(
 | 
				
			||||||
		users: (MiUser['id'] | MiUser)[],
 | 
							users: (MiUser['id'] | MiUser)[],
 | 
				
			||||||
		me?: { id: MiUser['id'] } | null | undefined,
 | 
							me?: { id: MiUser['id'] } | null | undefined,
 | 
				
			||||||
		options?: {
 | 
							options?: {
 | 
				
			||||||
| 
						 | 
					@ -560,6 +664,70 @@ export class UserEntityService implements OnModuleInit {
 | 
				
			||||||
			includeSecrets?: boolean,
 | 
								includeSecrets?: boolean,
 | 
				
			||||||
		},
 | 
							},
 | 
				
			||||||
	): Promise<Packed<S>[]> {
 | 
						): Promise<Packed<S>[]> {
 | 
				
			||||||
		return Promise.all(users.map(u => this.pack(u, me, options)));
 | 
							// -- IDのみの要素を補完して完全なエンティティ一覧を作る
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							const _users = users.filter((user): user is MiUser => typeof user !== 'string');
 | 
				
			||||||
 | 
							if (_users.length !== users.length) {
 | 
				
			||||||
 | 
								_users.push(
 | 
				
			||||||
 | 
									...await this.usersRepository.findBy({
 | 
				
			||||||
 | 
										id: In(users.filter((user): user is string => typeof user === 'string')),
 | 
				
			||||||
 | 
									}),
 | 
				
			||||||
 | 
								);
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							const _userIds = _users.map(u => u.id);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							// -- 特に前提条件のない値群を取得
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							const profilesMap = await this.userProfilesRepository.findBy({ userId: In(_userIds) })
 | 
				
			||||||
 | 
								.then(profiles => new Map(profiles.map(p => [p.userId, p])));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							// -- 実行者の有無や指定スキーマの種別によって要否が異なる値群を取得
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							let userRelations: Map<MiUser['id'], UserRelation> = new Map();
 | 
				
			||||||
 | 
							let userMemos: Map<MiUser['id'], string | null> = new Map();
 | 
				
			||||||
 | 
							let pinNotes: Map<MiUser['id'], MiUserNotePining[]> = new Map();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if (options?.schema !== 'UserLite') {
 | 
				
			||||||
 | 
								const meId = me ? me.id : null;
 | 
				
			||||||
 | 
								if (meId) {
 | 
				
			||||||
 | 
									userMemos = await this.userMemosRepository.findBy({ userId: meId })
 | 
				
			||||||
 | 
										.then(memos => new Map(memos.map(memo => [memo.targetUserId, memo.memo])));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
									if (_userIds.length > 0) {
 | 
				
			||||||
 | 
										userRelations = await this.getRelations(meId, _userIds);
 | 
				
			||||||
 | 
										pinNotes = await this.userNotePiningsRepository.createQueryBuilder('pin')
 | 
				
			||||||
 | 
											.where('pin.userId IN (:...userIds)', { userIds: _userIds })
 | 
				
			||||||
 | 
											.innerJoinAndSelect('pin.note', 'note')
 | 
				
			||||||
 | 
											.getMany()
 | 
				
			||||||
 | 
											.then(pinsNotes => {
 | 
				
			||||||
 | 
												const map = new Map<MiUser['id'], MiUserNotePining[]>();
 | 
				
			||||||
 | 
												for (const note of pinsNotes) {
 | 
				
			||||||
 | 
													const notes = map.get(note.userId) ?? [];
 | 
				
			||||||
 | 
													notes.push(note);
 | 
				
			||||||
 | 
													map.set(note.userId, notes);
 | 
				
			||||||
 | 
												}
 | 
				
			||||||
 | 
												for (const [, notes] of map.entries()) {
 | 
				
			||||||
 | 
													// pack側ではDESCで取得しているので、それに合わせて降順に並び替えておく
 | 
				
			||||||
 | 
													notes.sort((a, b) => b.id.localeCompare(a.id));
 | 
				
			||||||
 | 
												}
 | 
				
			||||||
 | 
												return map;
 | 
				
			||||||
 | 
											});
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							return Promise.all(
 | 
				
			||||||
 | 
								_users.map(u => this.pack(
 | 
				
			||||||
 | 
									u,
 | 
				
			||||||
 | 
									me,
 | 
				
			||||||
 | 
									{
 | 
				
			||||||
 | 
										...options,
 | 
				
			||||||
 | 
										userProfile: profilesMap.get(u.id),
 | 
				
			||||||
 | 
										userRelations: userRelations,
 | 
				
			||||||
 | 
										userMemos: userMemos,
 | 
				
			||||||
 | 
										pinNotes: pinNotes,
 | 
				
			||||||
 | 
									},
 | 
				
			||||||
 | 
								)),
 | 
				
			||||||
 | 
							);
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import type { onRequestHookHandler } from 'fastify';
 | 
					import type { onRequestHookHandler } from 'fastify';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export const handleRequestRedirectToOmitSearch: onRequestHookHandler = (request, reply, done) => {
 | 
					export const handleRequestRedirectToOmitSearch: onRequestHookHandler = (request, reply, done) => {
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import type { MiNote } from '@/models/Note.js';
 | 
					import type { MiNote } from '@/models/Note.js';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export function isPureRenote(note: MiNote): note is MiNote & { renoteId: NonNullable<MiNote['renoteId']> } {
 | 
					export function isPureRenote(note: MiNote): note is MiNote & { renoteId: NonNullable<MiNote['renoteId']> } {
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export type FetchFunction<K, V> = (key: K) => Promise<V>;
 | 
					export type FetchFunction<K, V> = (key: K) => Promise<V>;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
type ResolveReject<V> = Parameters<ConstructorParameters<typeof Promise<V>>[0]>;
 | 
					type ResolveReject<V> = Parameters<ConstructorParameters<typeof Promise<V>>[0]>;
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import { PrimaryColumn, Entity, Index, JoinColumn, Column, ManyToOne } from 'typeorm';
 | 
					import { PrimaryColumn, Entity, Index, JoinColumn, Column, ManyToOne } from 'typeorm';
 | 
				
			||||||
import { id } from './util/id.js';
 | 
					import { id } from './util/id.js';
 | 
				
			||||||
import { MiUser } from './User.js';
 | 
					import { MiUser } from './User.js';
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export const packedSigninSchema = {
 | 
					export const packedSigninSchema = {
 | 
				
			||||||
	type: 'object',
 | 
						type: 'object',
 | 
				
			||||||
	properties: {
 | 
						properties: {
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -67,7 +67,7 @@ export const paramDef = {
 | 
				
			||||||
		withFile: { type: 'boolean' },
 | 
							withFile: { type: 'boolean' },
 | 
				
			||||||
		notify: { type: 'boolean' },
 | 
							notify: { type: 'boolean' },
 | 
				
			||||||
	},
 | 
						},
 | 
				
			||||||
	required: ['antennaId', 'name', 'src', 'keywords', 'excludeKeywords', 'users', 'caseSensitive', 'withReplies', 'withFile', 'notify'],
 | 
						required: ['antennaId'],
 | 
				
			||||||
} as const;
 | 
					} as const;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@Injectable()
 | 
					@Injectable()
 | 
				
			||||||
| 
						 | 
					@ -83,9 +83,11 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint-
 | 
				
			||||||
		private globalEventService: GlobalEventService,
 | 
							private globalEventService: GlobalEventService,
 | 
				
			||||||
	) {
 | 
						) {
 | 
				
			||||||
		super(meta, paramDef, async (ps, me) => {
 | 
							super(meta, paramDef, async (ps, me) => {
 | 
				
			||||||
 | 
								if (ps.keywords && ps.excludeKeywords) {
 | 
				
			||||||
				if (ps.keywords.flat().every(x => x === '') && ps.excludeKeywords.flat().every(x => x === '')) {
 | 
									if (ps.keywords.flat().every(x => x === '') && ps.excludeKeywords.flat().every(x => x === '')) {
 | 
				
			||||||
					throw new Error('either keywords or excludeKeywords is required.');
 | 
										throw new Error('either keywords or excludeKeywords is required.');
 | 
				
			||||||
				}
 | 
									}
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
			// Fetch the antenna
 | 
								// Fetch the antenna
 | 
				
			||||||
			const antenna = await this.antennasRepository.findOneBy({
 | 
								const antenna = await this.antennasRepository.findOneBy({
 | 
				
			||||||
				id: ps.antennaId,
 | 
									id: ps.antennaId,
 | 
				
			||||||
| 
						 | 
					@ -98,7 +100,7 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint-
 | 
				
			||||||
 | 
					
 | 
				
			||||||
			let userList;
 | 
								let userList;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
			if (ps.src === 'list' && ps.userListId) {
 | 
								if ((ps.src === 'list' || antenna.src === 'list') && ps.userListId) {
 | 
				
			||||||
				userList = await this.userListsRepository.findOneBy({
 | 
									userList = await this.userListsRepository.findOneBy({
 | 
				
			||||||
					id: ps.userListId,
 | 
										id: ps.userListId,
 | 
				
			||||||
					userId: me.id,
 | 
										userId: me.id,
 | 
				
			||||||
| 
						 | 
					@ -112,7 +114,7 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint-
 | 
				
			||||||
			await this.antennasRepository.update(antenna.id, {
 | 
								await this.antennasRepository.update(antenna.id, {
 | 
				
			||||||
				name: ps.name,
 | 
									name: ps.name,
 | 
				
			||||||
				src: ps.src,
 | 
									src: ps.src,
 | 
				
			||||||
				userListId: userList ? userList.id : null,
 | 
									userListId: ps.userListId !== undefined ? userList ? userList.id : null : undefined,
 | 
				
			||||||
				keywords: ps.keywords,
 | 
									keywords: ps.keywords,
 | 
				
			||||||
				excludeKeywords: ps.excludeKeywords,
 | 
									excludeKeywords: ps.excludeKeywords,
 | 
				
			||||||
				users: ps.users,
 | 
									users: ps.users,
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -132,11 +132,9 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint-
 | 
				
			||||||
		private userEntityService: UserEntityService,
 | 
							private userEntityService: UserEntityService,
 | 
				
			||||||
	) {
 | 
						) {
 | 
				
			||||||
		super(meta, paramDef, async (ps, me) => {
 | 
							super(meta, paramDef, async (ps, me) => {
 | 
				
			||||||
			const ids = Array.isArray(ps.userId) ? ps.userId : [ps.userId];
 | 
								return Array.isArray(ps.userId)
 | 
				
			||||||
 | 
									? await this.userEntityService.getRelations(me.id, ps.userId).then(it => [...it.values()])
 | 
				
			||||||
			const relations = await Promise.all(ids.map(id => this.userEntityService.getRelation(me.id, id)));
 | 
									: await this.userEntityService.getRelation(me.id, ps.userId).then(it => [it]);
 | 
				
			||||||
 | 
					 | 
				
			||||||
			return Array.isArray(ps.userId) ? relations : relations[0];
 | 
					 | 
				
			||||||
		});
 | 
							});
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import { initTestDb, sendEnvResetRequest } from './utils.js';
 | 
					import { initTestDb, sendEnvResetRequest } from './utils.js';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
beforeAll(async () => {
 | 
					beforeAll(async () => {
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import * as assert from 'assert';
 | 
					import * as assert from 'assert';
 | 
				
			||||||
import { Test } from '@nestjs/testing';
 | 
					import { Test } from '@nestjs/testing';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
							
								
								
									
										528
									
								
								packages/backend/test/unit/entities/UserEntityService.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										528
									
								
								packages/backend/test/unit/entities/UserEntityService.ts
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
					@ -0,0 +1,528 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import { Test, TestingModule } from '@nestjs/testing';
 | 
				
			||||||
 | 
					import { UserEntityService } from '@/core/entities/UserEntityService.js';
 | 
				
			||||||
 | 
					import { GlobalModule } from '@/GlobalModule.js';
 | 
				
			||||||
 | 
					import { CoreModule } from '@/core/CoreModule.js';
 | 
				
			||||||
 | 
					import type { MiUser } from '@/models/User.js';
 | 
				
			||||||
 | 
					import { secureRndstr } from '@/misc/secure-rndstr.js';
 | 
				
			||||||
 | 
					import { genAidx } from '@/misc/id/aidx.js';
 | 
				
			||||||
 | 
					import {
 | 
				
			||||||
 | 
						BlockingsRepository,
 | 
				
			||||||
 | 
						FollowingsRepository, FollowRequestsRepository,
 | 
				
			||||||
 | 
						MiUserProfile, MutingsRepository, RenoteMutingsRepository,
 | 
				
			||||||
 | 
						UserMemoRepository,
 | 
				
			||||||
 | 
						UserProfilesRepository,
 | 
				
			||||||
 | 
						UsersRepository,
 | 
				
			||||||
 | 
					} from '@/models/_.js';
 | 
				
			||||||
 | 
					import { DI } from '@/di-symbols.js';
 | 
				
			||||||
 | 
					import { AvatarDecorationService } from '@/core/AvatarDecorationService.js';
 | 
				
			||||||
 | 
					import { ApPersonService } from '@/core/activitypub/models/ApPersonService.js';
 | 
				
			||||||
 | 
					import { NoteEntityService } from '@/core/entities/NoteEntityService.js';
 | 
				
			||||||
 | 
					import { PageEntityService } from '@/core/entities/PageEntityService.js';
 | 
				
			||||||
 | 
					import { CustomEmojiService } from '@/core/CustomEmojiService.js';
 | 
				
			||||||
 | 
					import { AnnouncementService } from '@/core/AnnouncementService.js';
 | 
				
			||||||
 | 
					import { RoleService } from '@/core/RoleService.js';
 | 
				
			||||||
 | 
					import { FederatedInstanceService } from '@/core/FederatedInstanceService.js';
 | 
				
			||||||
 | 
					import { IdService } from '@/core/IdService.js';
 | 
				
			||||||
 | 
					import { UtilityService } from '@/core/UtilityService.js';
 | 
				
			||||||
 | 
					import { EmojiEntityService } from '@/core/entities/EmojiEntityService.js';
 | 
				
			||||||
 | 
					import { ModerationLogService } from '@/core/ModerationLogService.js';
 | 
				
			||||||
 | 
					import { GlobalEventService } from '@/core/GlobalEventService.js';
 | 
				
			||||||
 | 
					import { DriveFileEntityService } from '@/core/entities/DriveFileEntityService.js';
 | 
				
			||||||
 | 
					import { MetaService } from '@/core/MetaService.js';
 | 
				
			||||||
 | 
					import { FetchInstanceMetadataService } from '@/core/FetchInstanceMetadataService.js';
 | 
				
			||||||
 | 
					import { CacheService } from '@/core/CacheService.js';
 | 
				
			||||||
 | 
					import { ApResolverService } from '@/core/activitypub/ApResolverService.js';
 | 
				
			||||||
 | 
					import { ApNoteService } from '@/core/activitypub/models/ApNoteService.js';
 | 
				
			||||||
 | 
					import { ApImageService } from '@/core/activitypub/models/ApImageService.js';
 | 
				
			||||||
 | 
					import { ApMfmService } from '@/core/activitypub/ApMfmService.js';
 | 
				
			||||||
 | 
					import { MfmService } from '@/core/MfmService.js';
 | 
				
			||||||
 | 
					import { HashtagService } from '@/core/HashtagService.js';
 | 
				
			||||||
 | 
					import UsersChart from '@/core/chart/charts/users.js';
 | 
				
			||||||
 | 
					import { ChartLoggerService } from '@/core/chart/ChartLoggerService.js';
 | 
				
			||||||
 | 
					import InstanceChart from '@/core/chart/charts/instance.js';
 | 
				
			||||||
 | 
					import { ApLoggerService } from '@/core/activitypub/ApLoggerService.js';
 | 
				
			||||||
 | 
					import { AccountMoveService } from '@/core/AccountMoveService.js';
 | 
				
			||||||
 | 
					import { ReactionService } from '@/core/ReactionService.js';
 | 
				
			||||||
 | 
					import { NotificationService } from '@/core/NotificationService.js';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					process.env.NODE_ENV = 'test';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					describe('UserEntityService', () => {
 | 
				
			||||||
 | 
						describe('pack/packMany', () => {
 | 
				
			||||||
 | 
							let app: TestingModule;
 | 
				
			||||||
 | 
							let service: UserEntityService;
 | 
				
			||||||
 | 
							let usersRepository: UsersRepository;
 | 
				
			||||||
 | 
							let userProfileRepository: UserProfilesRepository;
 | 
				
			||||||
 | 
							let userMemosRepository: UserMemoRepository;
 | 
				
			||||||
 | 
							let followingRepository: FollowingsRepository;
 | 
				
			||||||
 | 
							let followingRequestRepository: FollowRequestsRepository;
 | 
				
			||||||
 | 
							let blockingRepository: BlockingsRepository;
 | 
				
			||||||
 | 
							let mutingRepository: MutingsRepository;
 | 
				
			||||||
 | 
							let renoteMutingsRepository: RenoteMutingsRepository;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							async function createUser(userData: Partial<MiUser> = {}, profileData: Partial<MiUserProfile> = {}) {
 | 
				
			||||||
 | 
								const un = secureRndstr(16);
 | 
				
			||||||
 | 
								const user = await usersRepository
 | 
				
			||||||
 | 
									.insert({
 | 
				
			||||||
 | 
										...userData,
 | 
				
			||||||
 | 
										id: genAidx(Date.now()),
 | 
				
			||||||
 | 
										username: un,
 | 
				
			||||||
 | 
										usernameLower: un,
 | 
				
			||||||
 | 
									})
 | 
				
			||||||
 | 
									.then(x => usersRepository.findOneByOrFail(x.identifiers[0]));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								await userProfileRepository.insert({
 | 
				
			||||||
 | 
									...profileData,
 | 
				
			||||||
 | 
									userId: user.id,
 | 
				
			||||||
 | 
								});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								return user;
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							async function memo(writer: MiUser, target: MiUser, memo: string) {
 | 
				
			||||||
 | 
								await userMemosRepository.insert({
 | 
				
			||||||
 | 
									id: genAidx(Date.now()),
 | 
				
			||||||
 | 
									userId: writer.id,
 | 
				
			||||||
 | 
									targetUserId: target.id,
 | 
				
			||||||
 | 
									memo,
 | 
				
			||||||
 | 
								});
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							async function follow(follower: MiUser, followee: MiUser) {
 | 
				
			||||||
 | 
								await followingRepository.insert({
 | 
				
			||||||
 | 
									id: genAidx(Date.now()),
 | 
				
			||||||
 | 
									followerId: follower.id,
 | 
				
			||||||
 | 
									followeeId: followee.id,
 | 
				
			||||||
 | 
								});
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							async function requestFollow(requester: MiUser, requestee: MiUser) {
 | 
				
			||||||
 | 
								await followingRequestRepository.insert({
 | 
				
			||||||
 | 
									id: genAidx(Date.now()),
 | 
				
			||||||
 | 
									followerId: requester.id,
 | 
				
			||||||
 | 
									followeeId: requestee.id,
 | 
				
			||||||
 | 
								});
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							async function block(blocker: MiUser, blockee: MiUser) {
 | 
				
			||||||
 | 
								await blockingRepository.insert({
 | 
				
			||||||
 | 
									id: genAidx(Date.now()),
 | 
				
			||||||
 | 
									blockerId: blocker.id,
 | 
				
			||||||
 | 
									blockeeId: blockee.id,
 | 
				
			||||||
 | 
								});
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							async function mute(mutant: MiUser, mutee: MiUser) {
 | 
				
			||||||
 | 
								await mutingRepository.insert({
 | 
				
			||||||
 | 
									id: genAidx(Date.now()),
 | 
				
			||||||
 | 
									muterId: mutant.id,
 | 
				
			||||||
 | 
									muteeId: mutee.id,
 | 
				
			||||||
 | 
								});
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							async function muteRenote(mutant: MiUser, mutee: MiUser) {
 | 
				
			||||||
 | 
								await renoteMutingsRepository.insert({
 | 
				
			||||||
 | 
									id: genAidx(Date.now()),
 | 
				
			||||||
 | 
									muterId: mutant.id,
 | 
				
			||||||
 | 
									muteeId: mutee.id,
 | 
				
			||||||
 | 
								});
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							function randomIntRange(weight = 10) {
 | 
				
			||||||
 | 
								return [...Array(Math.floor(Math.random() * weight))].map((it, idx) => idx);
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							beforeAll(async () => {
 | 
				
			||||||
 | 
								const services = [
 | 
				
			||||||
 | 
									UserEntityService,
 | 
				
			||||||
 | 
									ApPersonService,
 | 
				
			||||||
 | 
									NoteEntityService,
 | 
				
			||||||
 | 
									PageEntityService,
 | 
				
			||||||
 | 
									CustomEmojiService,
 | 
				
			||||||
 | 
									AnnouncementService,
 | 
				
			||||||
 | 
									RoleService,
 | 
				
			||||||
 | 
									FederatedInstanceService,
 | 
				
			||||||
 | 
									IdService,
 | 
				
			||||||
 | 
									AvatarDecorationService,
 | 
				
			||||||
 | 
									UtilityService,
 | 
				
			||||||
 | 
									EmojiEntityService,
 | 
				
			||||||
 | 
									ModerationLogService,
 | 
				
			||||||
 | 
									GlobalEventService,
 | 
				
			||||||
 | 
									DriveFileEntityService,
 | 
				
			||||||
 | 
									MetaService,
 | 
				
			||||||
 | 
									FetchInstanceMetadataService,
 | 
				
			||||||
 | 
									CacheService,
 | 
				
			||||||
 | 
									ApResolverService,
 | 
				
			||||||
 | 
									ApNoteService,
 | 
				
			||||||
 | 
									ApImageService,
 | 
				
			||||||
 | 
									ApMfmService,
 | 
				
			||||||
 | 
									MfmService,
 | 
				
			||||||
 | 
									HashtagService,
 | 
				
			||||||
 | 
									UsersChart,
 | 
				
			||||||
 | 
									ChartLoggerService,
 | 
				
			||||||
 | 
									InstanceChart,
 | 
				
			||||||
 | 
									ApLoggerService,
 | 
				
			||||||
 | 
									AccountMoveService,
 | 
				
			||||||
 | 
									ReactionService,
 | 
				
			||||||
 | 
									NotificationService,
 | 
				
			||||||
 | 
								];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								app = await Test.createTestingModule({
 | 
				
			||||||
 | 
									imports: [GlobalModule, CoreModule],
 | 
				
			||||||
 | 
									providers: [
 | 
				
			||||||
 | 
										...services,
 | 
				
			||||||
 | 
										...services.map(x => ({ provide: x.name, useExisting: x })),
 | 
				
			||||||
 | 
									],
 | 
				
			||||||
 | 
								}).compile();
 | 
				
			||||||
 | 
								await app.init();
 | 
				
			||||||
 | 
								app.enableShutdownHooks();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								service = app.get<UserEntityService>(UserEntityService);
 | 
				
			||||||
 | 
								usersRepository = app.get<UsersRepository>(DI.usersRepository);
 | 
				
			||||||
 | 
								userProfileRepository = app.get<UserProfilesRepository>(DI.userProfilesRepository);
 | 
				
			||||||
 | 
								userMemosRepository = app.get<UserMemoRepository>(DI.userMemosRepository);
 | 
				
			||||||
 | 
								followingRepository = app.get<FollowingsRepository>(DI.followingsRepository);
 | 
				
			||||||
 | 
								followingRequestRepository = app.get<FollowRequestsRepository>(DI.followRequestsRepository);
 | 
				
			||||||
 | 
								blockingRepository = app.get<BlockingsRepository>(DI.blockingsRepository);
 | 
				
			||||||
 | 
								mutingRepository = app.get<MutingsRepository>(DI.mutingsRepository);
 | 
				
			||||||
 | 
								renoteMutingsRepository = app.get<RenoteMutingsRepository>(DI.renoteMutingsRepository);
 | 
				
			||||||
 | 
							});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							afterAll(async () => {
 | 
				
			||||||
 | 
								await app.close();
 | 
				
			||||||
 | 
							});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							test('UserLite', async() => {
 | 
				
			||||||
 | 
								const me = await createUser();
 | 
				
			||||||
 | 
								const who = await createUser();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								await memo(me, who, 'memo');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								const actual = await service.pack(who, me, { schema: 'UserLite' }) as any;
 | 
				
			||||||
 | 
								// no detail
 | 
				
			||||||
 | 
								expect(actual.memo).toBeUndefined();
 | 
				
			||||||
 | 
								// no detail and me
 | 
				
			||||||
 | 
								expect(actual.birthday).toBeUndefined();
 | 
				
			||||||
 | 
								// no detail and me
 | 
				
			||||||
 | 
								expect(actual.achievements).toBeUndefined();
 | 
				
			||||||
 | 
							});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							test('UserDetailedNotMe', async() => {
 | 
				
			||||||
 | 
								const me = await createUser();
 | 
				
			||||||
 | 
								const who = await createUser({}, { birthday: '2000-01-01' });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								await memo(me, who, 'memo');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								const actual = await service.pack(who, me, { schema: 'UserDetailedNotMe' }) as any;
 | 
				
			||||||
 | 
								// is detail
 | 
				
			||||||
 | 
								expect(actual.memo).toBe('memo');
 | 
				
			||||||
 | 
								// is detail
 | 
				
			||||||
 | 
								expect(actual.birthday).toBe('2000-01-01');
 | 
				
			||||||
 | 
								// no detail and me
 | 
				
			||||||
 | 
								expect(actual.achievements).toBeUndefined();
 | 
				
			||||||
 | 
							});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							test('MeDetailed', async() => {
 | 
				
			||||||
 | 
								const achievements = [{ name: 'achievement', unlockedAt: new Date().getTime() }];
 | 
				
			||||||
 | 
								const me = await createUser({}, {
 | 
				
			||||||
 | 
									birthday: '2000-01-01',
 | 
				
			||||||
 | 
									achievements: achievements,
 | 
				
			||||||
 | 
								});
 | 
				
			||||||
 | 
								await memo(me, me, 'memo');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								const actual = await service.pack(me, me, { schema: 'MeDetailed' }) as any;
 | 
				
			||||||
 | 
								// is detail
 | 
				
			||||||
 | 
								expect(actual.memo).toBe('memo');
 | 
				
			||||||
 | 
								// is detail
 | 
				
			||||||
 | 
								expect(actual.birthday).toBe('2000-01-01');
 | 
				
			||||||
 | 
								// is detail and me
 | 
				
			||||||
 | 
								expect(actual.achievements).toEqual(achievements);
 | 
				
			||||||
 | 
							});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							describe('packManyによるpreloadがある時、preloadが無い時とpackの結果が同じになるか見たい', () => {
 | 
				
			||||||
 | 
								test('no-preload', async() => {
 | 
				
			||||||
 | 
									const me = await createUser();
 | 
				
			||||||
 | 
									// meがフォローしてる人たち
 | 
				
			||||||
 | 
									const followeeMe = await Promise.all(randomIntRange().map(() => createUser()));
 | 
				
			||||||
 | 
									for (const who of followeeMe) {
 | 
				
			||||||
 | 
										await follow(me, who);
 | 
				
			||||||
 | 
										const actual = await service.pack(who, me, { schema: 'UserDetailed' }) as any;
 | 
				
			||||||
 | 
										expect(actual.isFollowing).toBe(true);
 | 
				
			||||||
 | 
										expect(actual.isFollowed).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.hasPendingFollowRequestFromYou).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.hasPendingFollowRequestToYou).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isBlocking).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isBlocked).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isMuted).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isRenoteMuted).toBe(false);
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
									// meをフォローしてる人たち
 | 
				
			||||||
 | 
									const followerMe = await Promise.all(randomIntRange().map(() => createUser()));
 | 
				
			||||||
 | 
									for (const who of followerMe) {
 | 
				
			||||||
 | 
										await follow(who, me);
 | 
				
			||||||
 | 
										const actual = await service.pack(who, me, { schema: 'UserDetailed' }) as any;
 | 
				
			||||||
 | 
										expect(actual.isFollowing).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isFollowed).toBe(true);
 | 
				
			||||||
 | 
										expect(actual.hasPendingFollowRequestFromYou).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.hasPendingFollowRequestToYou).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isBlocking).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isBlocked).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isMuted).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isRenoteMuted).toBe(false);
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
									// meがフォローリクエストを送った人たち
 | 
				
			||||||
 | 
									const requestsFromYou = await Promise.all(randomIntRange().map(() => createUser()));
 | 
				
			||||||
 | 
									for (const who of requestsFromYou) {
 | 
				
			||||||
 | 
										await requestFollow(me, who);
 | 
				
			||||||
 | 
										const actual = await service.pack(who, me, { schema: 'UserDetailed' }) as any;
 | 
				
			||||||
 | 
										expect(actual.isFollowing).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isFollowed).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.hasPendingFollowRequestFromYou).toBe(true);
 | 
				
			||||||
 | 
										expect(actual.hasPendingFollowRequestToYou).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isBlocking).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isBlocked).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isMuted).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isRenoteMuted).toBe(false);
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
									// meにフォローリクエストを送った人たち
 | 
				
			||||||
 | 
									const requestsToYou = await Promise.all(randomIntRange().map(() => createUser()));
 | 
				
			||||||
 | 
									for (const who of requestsToYou) {
 | 
				
			||||||
 | 
										await requestFollow(who, me);
 | 
				
			||||||
 | 
										const actual = await service.pack(who, me, { schema: 'UserDetailed' }) as any;
 | 
				
			||||||
 | 
										expect(actual.isFollowing).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isFollowed).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.hasPendingFollowRequestFromYou).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.hasPendingFollowRequestToYou).toBe(true);
 | 
				
			||||||
 | 
										expect(actual.isBlocking).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isBlocked).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isMuted).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isRenoteMuted).toBe(false);
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
									// meがブロックしてる人たち
 | 
				
			||||||
 | 
									const blockingYou = await Promise.all(randomIntRange().map(() => createUser()));
 | 
				
			||||||
 | 
									for (const who of blockingYou) {
 | 
				
			||||||
 | 
										await block(me, who);
 | 
				
			||||||
 | 
										const actual = await service.pack(who, me, { schema: 'UserDetailed' }) as any;
 | 
				
			||||||
 | 
										expect(actual.isFollowing).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isFollowed).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.hasPendingFollowRequestFromYou).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.hasPendingFollowRequestToYou).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isBlocking).toBe(true);
 | 
				
			||||||
 | 
										expect(actual.isBlocked).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isMuted).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isRenoteMuted).toBe(false);
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
									// meをブロックしてる人たち
 | 
				
			||||||
 | 
									const blockingMe = await Promise.all(randomIntRange().map(() => createUser()));
 | 
				
			||||||
 | 
									for (const who of blockingMe) {
 | 
				
			||||||
 | 
										await block(who, me);
 | 
				
			||||||
 | 
										const actual = await service.pack(who, me, { schema: 'UserDetailed' }) as any;
 | 
				
			||||||
 | 
										expect(actual.isFollowing).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isFollowed).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.hasPendingFollowRequestFromYou).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.hasPendingFollowRequestToYou).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isBlocking).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isBlocked).toBe(true);
 | 
				
			||||||
 | 
										expect(actual.isMuted).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isRenoteMuted).toBe(false);
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
									// meがミュートしてる人たち
 | 
				
			||||||
 | 
									const muters = await Promise.all(randomIntRange().map(() => createUser()));
 | 
				
			||||||
 | 
									for (const who of muters) {
 | 
				
			||||||
 | 
										await mute(me, who);
 | 
				
			||||||
 | 
										const actual = await service.pack(who, me, { schema: 'UserDetailed' }) as any;
 | 
				
			||||||
 | 
										expect(actual.isFollowing).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isFollowed).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.hasPendingFollowRequestFromYou).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.hasPendingFollowRequestToYou).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isBlocking).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isBlocked).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isMuted).toBe(true);
 | 
				
			||||||
 | 
										expect(actual.isRenoteMuted).toBe(false);
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
									// meがリノートミュートしてる人たち
 | 
				
			||||||
 | 
									const renoteMuters = await Promise.all(randomIntRange().map(() => createUser()));
 | 
				
			||||||
 | 
									for (const who of renoteMuters) {
 | 
				
			||||||
 | 
										await muteRenote(me, who);
 | 
				
			||||||
 | 
										const actual = await service.pack(who, me, { schema: 'UserDetailed' }) as any;
 | 
				
			||||||
 | 
										expect(actual.isFollowing).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isFollowed).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.hasPendingFollowRequestFromYou).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.hasPendingFollowRequestToYou).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isBlocking).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isBlocked).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isMuted).toBe(false);
 | 
				
			||||||
 | 
										expect(actual.isRenoteMuted).toBe(true);
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
								});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								test('preload', async() => {
 | 
				
			||||||
 | 
									const me = await createUser();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
									{
 | 
				
			||||||
 | 
										// meがフォローしてる人たち
 | 
				
			||||||
 | 
										const followeeMe = await Promise.all(randomIntRange().map(() => createUser()));
 | 
				
			||||||
 | 
										for (const who of followeeMe) {
 | 
				
			||||||
 | 
											await follow(me, who);
 | 
				
			||||||
 | 
										}
 | 
				
			||||||
 | 
										const actualList = await service.packMany(followeeMe, me, { schema: 'UserDetailed' }) as any;
 | 
				
			||||||
 | 
										for (const actual of actualList) {
 | 
				
			||||||
 | 
											expect(actual.isFollowing).toBe(true);
 | 
				
			||||||
 | 
											expect(actual.isFollowed).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.hasPendingFollowRequestFromYou).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.hasPendingFollowRequestToYou).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isBlocking).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isBlocked).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isMuted).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isRenoteMuted).toBe(false);
 | 
				
			||||||
 | 
										}
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
									{
 | 
				
			||||||
 | 
										// meをフォローしてる人たち
 | 
				
			||||||
 | 
										const followerMe = await Promise.all(randomIntRange().map(() => createUser()));
 | 
				
			||||||
 | 
										for (const who of followerMe) {
 | 
				
			||||||
 | 
											await follow(who, me);
 | 
				
			||||||
 | 
										}
 | 
				
			||||||
 | 
										const actualList = await service.packMany(followerMe, me, { schema: 'UserDetailed' }) as any;
 | 
				
			||||||
 | 
										for (const actual of actualList) {
 | 
				
			||||||
 | 
											expect(actual.isFollowing).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isFollowed).toBe(true);
 | 
				
			||||||
 | 
											expect(actual.hasPendingFollowRequestFromYou).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.hasPendingFollowRequestToYou).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isBlocking).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isBlocked).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isMuted).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isRenoteMuted).toBe(false);
 | 
				
			||||||
 | 
										}
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
									{
 | 
				
			||||||
 | 
										// meがフォローリクエストを送った人たち
 | 
				
			||||||
 | 
										const requestsFromYou = await Promise.all(randomIntRange().map(() => createUser()));
 | 
				
			||||||
 | 
										for (const who of requestsFromYou) {
 | 
				
			||||||
 | 
											await requestFollow(me, who);
 | 
				
			||||||
 | 
										}
 | 
				
			||||||
 | 
										const actualList = await service.packMany(requestsFromYou, me, { schema: 'UserDetailed' }) as any;
 | 
				
			||||||
 | 
										for (const actual of actualList) {
 | 
				
			||||||
 | 
											expect(actual.isFollowing).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isFollowed).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.hasPendingFollowRequestFromYou).toBe(true);
 | 
				
			||||||
 | 
											expect(actual.hasPendingFollowRequestToYou).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isBlocking).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isBlocked).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isMuted).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isRenoteMuted).toBe(false);
 | 
				
			||||||
 | 
										}
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
									{
 | 
				
			||||||
 | 
										// meにフォローリクエストを送った人たち
 | 
				
			||||||
 | 
										const requestsToYou = await Promise.all(randomIntRange().map(() => createUser()));
 | 
				
			||||||
 | 
										for (const who of requestsToYou) {
 | 
				
			||||||
 | 
											await requestFollow(who, me);
 | 
				
			||||||
 | 
										}
 | 
				
			||||||
 | 
										const actualList = await service.packMany(requestsToYou, me, { schema: 'UserDetailed' }) as any;
 | 
				
			||||||
 | 
										for (const actual of actualList) {
 | 
				
			||||||
 | 
											expect(actual.isFollowing).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isFollowed).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.hasPendingFollowRequestFromYou).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.hasPendingFollowRequestToYou).toBe(true);
 | 
				
			||||||
 | 
											expect(actual.isBlocking).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isBlocked).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isMuted).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isRenoteMuted).toBe(false);
 | 
				
			||||||
 | 
										}
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
									{
 | 
				
			||||||
 | 
										// meがブロックしてる人たち
 | 
				
			||||||
 | 
										const blockingYou = await Promise.all(randomIntRange().map(() => createUser()));
 | 
				
			||||||
 | 
										for (const who of blockingYou) {
 | 
				
			||||||
 | 
											await block(me, who);
 | 
				
			||||||
 | 
										}
 | 
				
			||||||
 | 
										const actualList = await service.packMany(blockingYou, me, { schema: 'UserDetailed' }) as any;
 | 
				
			||||||
 | 
										for (const actual of actualList) {
 | 
				
			||||||
 | 
											expect(actual.isFollowing).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isFollowed).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.hasPendingFollowRequestFromYou).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.hasPendingFollowRequestToYou).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isBlocking).toBe(true);
 | 
				
			||||||
 | 
											expect(actual.isBlocked).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isMuted).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isRenoteMuted).toBe(false);
 | 
				
			||||||
 | 
										}
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
									{
 | 
				
			||||||
 | 
										// meをブロックしてる人たち
 | 
				
			||||||
 | 
										const blockingMe = await Promise.all(randomIntRange().map(() => createUser()));
 | 
				
			||||||
 | 
										for (const who of blockingMe) {
 | 
				
			||||||
 | 
											await block(who, me);
 | 
				
			||||||
 | 
										}
 | 
				
			||||||
 | 
										const actualList = await service.packMany(blockingMe, me, { schema: 'UserDetailed' }) as any;
 | 
				
			||||||
 | 
										for (const actual of actualList) {
 | 
				
			||||||
 | 
											expect(actual.isFollowing).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isFollowed).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.hasPendingFollowRequestFromYou).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.hasPendingFollowRequestToYou).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isBlocking).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isBlocked).toBe(true);
 | 
				
			||||||
 | 
											expect(actual.isMuted).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isRenoteMuted).toBe(false);
 | 
				
			||||||
 | 
										}
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
									{
 | 
				
			||||||
 | 
										// meがミュートしてる人たち
 | 
				
			||||||
 | 
										const muters = await Promise.all(randomIntRange().map(() => createUser()));
 | 
				
			||||||
 | 
										for (const who of muters) {
 | 
				
			||||||
 | 
											await mute(me, who);
 | 
				
			||||||
 | 
										}
 | 
				
			||||||
 | 
										const actualList = await service.packMany(muters, me, { schema: 'UserDetailed' }) as any;
 | 
				
			||||||
 | 
										for (const actual of actualList) {
 | 
				
			||||||
 | 
											expect(actual.isFollowing).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isFollowed).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.hasPendingFollowRequestFromYou).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.hasPendingFollowRequestToYou).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isBlocking).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isBlocked).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isMuted).toBe(true);
 | 
				
			||||||
 | 
											expect(actual.isRenoteMuted).toBe(false);
 | 
				
			||||||
 | 
										}
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
									{
 | 
				
			||||||
 | 
										// meがリノートミュートしてる人たち
 | 
				
			||||||
 | 
										const renoteMuters = await Promise.all(randomIntRange().map(() => createUser()));
 | 
				
			||||||
 | 
										for (const who of renoteMuters) {
 | 
				
			||||||
 | 
											await muteRenote(me, who);
 | 
				
			||||||
 | 
										}
 | 
				
			||||||
 | 
										const actualList = await service.packMany(renoteMuters, me, { schema: 'UserDetailed' }) as any;
 | 
				
			||||||
 | 
										for (const actual of actualList) {
 | 
				
			||||||
 | 
											expect(actual.isFollowing).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isFollowed).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.hasPendingFollowRequestFromYou).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.hasPendingFollowRequestToYou).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isBlocking).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isBlocked).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isMuted).toBe(false);
 | 
				
			||||||
 | 
											expect(actual.isRenoteMuted).toBe(true);
 | 
				
			||||||
 | 
										}
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
								});
 | 
				
			||||||
 | 
							});
 | 
				
			||||||
 | 
						});
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import { DebounceLoader } from '@/misc/loader.js';
 | 
					import { DebounceLoader } from '@/misc/loader.js';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class Mock {
 | 
					class Mock {
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					<!--
 | 
				
			||||||
 | 
					SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					-->
 | 
				
			||||||
 | 
					
 | 
				
			||||||
<link rel="preload" href="https://github.com/misskey-dev/misskey/blob/master/packages/frontend/assets/about-icon.png?raw=true" as="image" type="image/png" crossorigin="anonymous">
 | 
					<link rel="preload" href="https://github.com/misskey-dev/misskey/blob/master/packages/frontend/assets/about-icon.png?raw=true" as="image" type="image/png" crossorigin="anonymous">
 | 
				
			||||||
<link rel="preload" href="https://github.com/misskey-dev/misskey/blob/master/packages/frontend/assets/fedi.jpg?raw=true" as="image" type="image/jpeg" crossorigin="anonymous">
 | 
					<link rel="preload" href="https://github.com/misskey-dev/misskey/blob/master/packages/frontend/assets/fedi.jpg?raw=true" as="image" type="image/jpeg" crossorigin="anonymous">
 | 
				
			||||||
<link rel="stylesheet" href="https://unpkg.com/@tabler/icons-webfont@2.44.0/tabler-icons.min.css">
 | 
					<link rel="stylesheet" href="https://unpkg.com/@tabler/icons-webfont@2.44.0/tabler-icons.min.css">
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -30,13 +30,13 @@ const self = props.url.startsWith(local);
 | 
				
			||||||
const attr = self ? 'to' : 'href';
 | 
					const attr = self ? 'to' : 'href';
 | 
				
			||||||
const target = self ? null : '_blank';
 | 
					const target = self ? null : '_blank';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const el = ref<HTMLElement>();
 | 
					const el = ref<HTMLElement | { $el: HTMLElement }>();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
useTooltip(el, (showing) => {
 | 
					useTooltip(el, (showing) => {
 | 
				
			||||||
	os.popup(defineAsyncComponent(() => import('@/components/MkUrlPreviewPopup.vue')), {
 | 
						os.popup(defineAsyncComponent(() => import('@/components/MkUrlPreviewPopup.vue')), {
 | 
				
			||||||
		showing,
 | 
							showing,
 | 
				
			||||||
		url: props.url,
 | 
							url: props.url,
 | 
				
			||||||
		source: el.value,
 | 
							source: el.value instanceof HTMLElement ? el.value : el.value?.$el,
 | 
				
			||||||
	}, {}, 'closed');
 | 
						}, {}, 'closed');
 | 
				
			||||||
});
 | 
					});
 | 
				
			||||||
</script>
 | 
					</script>
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					<!--
 | 
				
			||||||
 | 
					SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					-->
 | 
				
			||||||
 | 
					
 | 
				
			||||||
<template>
 | 
					<template>
 | 
				
			||||||
<render/>
 | 
					<render/>
 | 
				
			||||||
</template>
 | 
					</template>
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -4,13 +4,13 @@ SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
-->
 | 
					-->
 | 
				
			||||||
 | 
					
 | 
				
			||||||
<template>
 | 
					<template>
 | 
				
			||||||
<a :href="to" :class="active ? activeClass : null" @click.prevent="nav" @contextmenu.prevent.stop="onContextmenu" @click.stop>
 | 
					<a ref="el" :href="to" :class="active ? activeClass : null" @click.prevent="nav" @contextmenu.prevent.stop="onContextmenu" @click.stop>
 | 
				
			||||||
	<slot></slot>
 | 
						<slot></slot>
 | 
				
			||||||
</a>
 | 
					</a>
 | 
				
			||||||
</template>
 | 
					</template>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
<script lang="ts" setup>
 | 
					<script lang="ts" setup>
 | 
				
			||||||
import { computed } from 'vue';
 | 
					import { computed, shallowRef } from 'vue';
 | 
				
			||||||
import * as os from '@/os.js';
 | 
					import * as os from '@/os.js';
 | 
				
			||||||
import copyToClipboard from '@/scripts/copy-to-clipboard.js';
 | 
					import copyToClipboard from '@/scripts/copy-to-clipboard.js';
 | 
				
			||||||
import { url } from '@/config.js';
 | 
					import { url } from '@/config.js';
 | 
				
			||||||
| 
						 | 
					@ -26,6 +26,10 @@ const props = withDefaults(defineProps<{
 | 
				
			||||||
	behavior: null,
 | 
						behavior: null,
 | 
				
			||||||
});
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const el = shallowRef<HTMLElement>();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					defineExpose({ $el: el });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const router = useRouter();
 | 
					const router = useRouter();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const active = computed(() => {
 | 
					const active = computed(() => {
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -50,7 +50,7 @@ if (props.showUrlPreview) {
 | 
				
			||||||
		os.popup(defineAsyncComponent(() => import('@/components/MkUrlPreviewPopup.vue')), {
 | 
							os.popup(defineAsyncComponent(() => import('@/components/MkUrlPreviewPopup.vue')), {
 | 
				
			||||||
			showing,
 | 
								showing,
 | 
				
			||||||
			url: props.url,
 | 
								url: props.url,
 | 
				
			||||||
			source: el.value,
 | 
								source: el.value instanceof HTMLElement ? el.value : el.value?.$el,
 | 
				
			||||||
		}, {}, 'closed');
 | 
							}, {}, 'closed');
 | 
				
			||||||
	});
 | 
						});
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export default (v, fractionDigits = 0) => {
 | 
					export default (v, fractionDigits = 0) => {
 | 
				
			||||||
	if (v == null) return 'N/A';
 | 
						if (v == null) return 'N/A';
 | 
				
			||||||
	if (v === 0) return '0';
 | 
						if (v === 0) return '0';
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -41,13 +41,26 @@ SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
					<MkButton inline danger @click="uninstall(plugin)"><i class="ph-trash ph-bold ph-lg"></i> {{ i18n.ts.uninstall }}</MkButton>
 | 
										<MkButton inline danger @click="uninstall(plugin)"><i class="ph-trash ph-bold ph-lg"></i> {{ i18n.ts.uninstall }}</MkButton>
 | 
				
			||||||
				</div>
 | 
									</div>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
									<MkFolder>
 | 
				
			||||||
 | 
										<template #icon><i class="ph-terminal-window ph-bold ph-lg"></i></template>
 | 
				
			||||||
 | 
										<template #label>{{ i18n.ts._plugin.viewLog }}</template>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
										<div class="_gaps_s">
 | 
				
			||||||
 | 
											<div class="_buttons">
 | 
				
			||||||
 | 
												<MkButton inline @click="copy(pluginLogs.get(plugin.id)?.join('\n'))"><i class="ti ti-copy"></i> {{ i18n.ts.copy }}</MkButton>
 | 
				
			||||||
 | 
											</div>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
											<MkCode :code="pluginLogs.get(plugin.id)?.join('\n') ?? ''"/>
 | 
				
			||||||
 | 
										</div>
 | 
				
			||||||
 | 
									</MkFolder>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
				<MkFolder>
 | 
									<MkFolder>
 | 
				
			||||||
					<template #icon><i class="ph-code ph-bold ph-lg"></i></template>
 | 
										<template #icon><i class="ph-code ph-bold ph-lg"></i></template>
 | 
				
			||||||
					<template #label>{{ i18n.ts._plugin.viewSource }}</template>
 | 
										<template #label>{{ i18n.ts._plugin.viewSource }}</template>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
					<div class="_gaps_s">
 | 
										<div class="_gaps_s">
 | 
				
			||||||
						<div class="_buttons">
 | 
											<div class="_buttons">
 | 
				
			||||||
							<MkButton inline @click="copy(plugin)"><i class="ph-copy ph-bold ph-lg"></i> {{ i18n.ts.copy }}</MkButton>
 | 
												<MkButton inline @click="copy(plugin.src)"><i class="ph-copy ph-bold ph-lg"></i> {{ i18n.ts.copy }}</MkButton>
 | 
				
			||||||
						</div>
 | 
											</div>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
						<MkCode :code="plugin.src ?? ''" lang="is"/>
 | 
											<MkCode :code="plugin.src ?? ''" lang="is"/>
 | 
				
			||||||
| 
						 | 
					@ -74,6 +87,7 @@ import { ColdDeviceStorage } from '@/store.js';
 | 
				
			||||||
import { unisonReload } from '@/scripts/unison-reload.js';
 | 
					import { unisonReload } from '@/scripts/unison-reload.js';
 | 
				
			||||||
import { i18n } from '@/i18n.js';
 | 
					import { i18n } from '@/i18n.js';
 | 
				
			||||||
import { definePageMetadata } from '@/scripts/page-metadata.js';
 | 
					import { definePageMetadata } from '@/scripts/page-metadata.js';
 | 
				
			||||||
 | 
					import { pluginLogs } from '@/plugin.js';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const plugins = ref(ColdDeviceStorage.get('plugins'));
 | 
					const plugins = ref(ColdDeviceStorage.get('plugins'));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -87,8 +101,8 @@ async function uninstall(plugin) {
 | 
				
			||||||
	});
 | 
						});
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function copy(plugin) {
 | 
					function copy(text) {
 | 
				
			||||||
	copyToClipboard(plugin.src ?? '');
 | 
						copyToClipboard(text ?? '');
 | 
				
			||||||
	os.success();
 | 
						os.success();
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -9,7 +9,10 @@ SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
	<XTimeline class="tl"/>
 | 
						<XTimeline class="tl"/>
 | 
				
			||||||
	<div class="shape1"></div>
 | 
						<div class="shape1"></div>
 | 
				
			||||||
	<div class="shape2"></div>
 | 
						<div class="shape2"></div>
 | 
				
			||||||
 | 
						<div class="logo-wrapper">
 | 
				
			||||||
 | 
							<div class="powered-by">Powered by</div>
 | 
				
			||||||
		<img :src="misskeysvg" class="misskey"/>
 | 
							<img :src="misskeysvg" class="misskey"/>
 | 
				
			||||||
 | 
						</div>
 | 
				
			||||||
	<div class="emojis">
 | 
						<div class="emojis">
 | 
				
			||||||
		<MkEmoji :normal="true" :noStyle="true" emoji="👍"/>
 | 
							<MkEmoji :normal="true" :noStyle="true" emoji="👍"/>
 | 
				
			||||||
		<MkEmoji :normal="true" :noStyle="true" emoji="❤"/>
 | 
							<MkEmoji :normal="true" :noStyle="true" emoji="❤"/>
 | 
				
			||||||
| 
						 | 
					@ -113,16 +116,26 @@ misskeyApiGet('federation/instances', {
 | 
				
			||||||
		opacity: 0.5;
 | 
							opacity: 0.5;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	> .misskey {
 | 
						> .logo-wrapper {
 | 
				
			||||||
		position: fixed;
 | 
							position: fixed;
 | 
				
			||||||
		top: 42px;
 | 
							top: 36px;
 | 
				
			||||||
		left: 42px;
 | 
							left: 36px;
 | 
				
			||||||
		width: 140px;
 | 
							flex: auto;
 | 
				
			||||||
 | 
							color: #fff;
 | 
				
			||||||
 | 
							user-select: none;
 | 
				
			||||||
 | 
							pointer-events: none;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							> .powered-by {
 | 
				
			||||||
 | 
								margin-bottom: 2px;
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							> .misskey {
 | 
				
			||||||
 | 
								width: 140px;
 | 
				
			||||||
			@media (max-width: 450px) {
 | 
								@media (max-width: 450px) {
 | 
				
			||||||
				width: 130px;
 | 
									width: 130px;
 | 
				
			||||||
			}
 | 
								}
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	> .emojis {
 | 
						> .emojis {
 | 
				
			||||||
		position: fixed;
 | 
							position: fixed;
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -3,6 +3,7 @@
 | 
				
			||||||
 * SPDX-License-Identifier: AGPL-3.0-only
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 */
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import { ref } from 'vue';
 | 
				
			||||||
import { Interpreter, Parser, utils, values } from '@syuilo/aiscript';
 | 
					import { Interpreter, Parser, utils, values } from '@syuilo/aiscript';
 | 
				
			||||||
import { aiScriptReadline, createAiScriptEnv } from '@/scripts/aiscript/api.js';
 | 
					import { aiScriptReadline, createAiScriptEnv } from '@/scripts/aiscript/api.js';
 | 
				
			||||||
import { inputText } from '@/os.js';
 | 
					import { inputText } from '@/os.js';
 | 
				
			||||||
| 
						 | 
					@ -10,6 +11,7 @@ import { Plugin, noteActions, notePostInterruptors, noteViewInterruptors, postFo
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const parser = new Parser();
 | 
					const parser = new Parser();
 | 
				
			||||||
const pluginContexts = new Map<string, Interpreter>();
 | 
					const pluginContexts = new Map<string, Interpreter>();
 | 
				
			||||||
 | 
					export const pluginLogs = ref(new Map<string, string[]>());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export async function install(plugin: Plugin): Promise<void> {
 | 
					export async function install(plugin: Plugin): Promise<void> {
 | 
				
			||||||
	// 後方互換性のため
 | 
						// 後方互換性のため
 | 
				
			||||||
| 
						 | 
					@ -22,21 +24,27 @@ export async function install(plugin: Plugin): Promise<void> {
 | 
				
			||||||
		in: aiScriptReadline,
 | 
							in: aiScriptReadline,
 | 
				
			||||||
		out: (value): void => {
 | 
							out: (value): void => {
 | 
				
			||||||
			console.log(value);
 | 
								console.log(value);
 | 
				
			||||||
 | 
								pluginLogs.value.get(plugin.id).push(utils.reprValue(value));
 | 
				
			||||||
		},
 | 
							},
 | 
				
			||||||
		log: (): void => {
 | 
							log: (): void => {
 | 
				
			||||||
		},
 | 
							},
 | 
				
			||||||
 | 
							err: (err): void => {
 | 
				
			||||||
 | 
								pluginLogs.value.get(plugin.id).push(`${err}`);
 | 
				
			||||||
 | 
								throw err; // install時のtry-catchに反応させる
 | 
				
			||||||
 | 
							},
 | 
				
			||||||
	});
 | 
						});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	initPlugin({ plugin, aiscript });
 | 
						initPlugin({ plugin, aiscript });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	try {
 | 
						aiscript.exec(parser.parse(plugin.src)).then(
 | 
				
			||||||
		await aiscript.exec(parser.parse(plugin.src));
 | 
							() => {
 | 
				
			||||||
	} catch (err) {
 | 
					 | 
				
			||||||
		console.error('Plugin install failed:', plugin.name, 'v' + plugin.version);
 | 
					 | 
				
			||||||
		return;
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
			console.info('Plugin installed:', plugin.name, 'v' + plugin.version);
 | 
								console.info('Plugin installed:', plugin.name, 'v' + plugin.version);
 | 
				
			||||||
 | 
							},
 | 
				
			||||||
 | 
							(err) => {
 | 
				
			||||||
 | 
								console.error('Plugin install failed:', plugin.name, 'v' + plugin.version);
 | 
				
			||||||
 | 
								throw err;
 | 
				
			||||||
 | 
							},
 | 
				
			||||||
 | 
						);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function createPluginEnv(opts: { plugin: Plugin; storageKey: string }): Record<string, values.Value> {
 | 
					function createPluginEnv(opts: { plugin: Plugin; storageKey: string }): Record<string, values.Value> {
 | 
				
			||||||
| 
						 | 
					@ -92,6 +100,7 @@ function createPluginEnv(opts: { plugin: Plugin; storageKey: string }): Record<s
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function initPlugin({ plugin, aiscript }): void {
 | 
					function initPlugin({ plugin, aiscript }): void {
 | 
				
			||||||
	pluginContexts.set(plugin.id, aiscript);
 | 
						pluginContexts.set(plugin.id, aiscript);
 | 
				
			||||||
 | 
						pluginLogs.value.set(plugin.id, []);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
function registerPostFormAction({ pluginId, title, handler }): void {
 | 
					function registerPostFormAction({ pluginId, title, handler }): void {
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import * as Misskey from 'misskey-js';
 | 
					import * as Misskey from 'misskey-js';
 | 
				
			||||||
import { UnicodeEmojiDef } from './emojilist.js';
 | 
					import { UnicodeEmojiDef } from './emojilist.js';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import { unisonReload } from '@/scripts/unison-reload.js';
 | 
					import { unisonReload } from '@/scripts/unison-reload.js';
 | 
				
			||||||
import * as os from '@/os.js';
 | 
					import * as os from '@/os.js';
 | 
				
			||||||
import { miLocalStorage } from '@/local-storage.js';
 | 
					import { miLocalStorage } from '@/local-storage.js';
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import { bundledThemesInfo } from 'shiki';
 | 
					import { bundledThemesInfo } from 'shiki';
 | 
				
			||||||
import { getHighlighterCore, loadWasm } from 'shiki/core';
 | 
					import { getHighlighterCore, loadWasm } from 'shiki/core';
 | 
				
			||||||
import darkPlus from 'shiki/themes/dark-plus.mjs';
 | 
					import darkPlus from 'shiki/themes/dark-plus.mjs';
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export default async function hasAudio(media: HTMLMediaElement) {
 | 
					export default async function hasAudio(media: HTMLMediaElement) {
 | 
				
			||||||
	const cloned = media.cloneNode() as HTMLMediaElement;
 | 
						const cloned = media.cloneNode() as HTMLMediaElement;
 | 
				
			||||||
	cloned.muted = (cloned as typeof cloned & Partial<HTMLVideoElement>).playsInline = true;
 | 
						cloned.muted = (cloned as typeof cloned & Partial<HTMLVideoElement>).playsInline = true;
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export type WithRequired<T, K extends keyof T> = T & { [P in K]-?: T[P] };
 | 
					export type WithRequired<T, K extends keyof T> = T & { [P in K]-?: T[P] };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export type WithNonNullable<T, K extends keyof T> = T & { [P in K]-?: NonNullable<T[P]> };
 | 
					export type WithNonNullable<T, K extends keyof T> = T & { [P in K]-?: NonNullable<T[P]> };
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -10350,19 +10350,19 @@ export type operations = {
 | 
				
			||||||
        'application/json': {
 | 
					        'application/json': {
 | 
				
			||||||
          /** Format: misskey:id */
 | 
					          /** Format: misskey:id */
 | 
				
			||||||
          antennaId: string;
 | 
					          antennaId: string;
 | 
				
			||||||
          name: string;
 | 
					          name?: string;
 | 
				
			||||||
          /** @enum {string} */
 | 
					          /** @enum {string} */
 | 
				
			||||||
          src: 'home' | 'all' | 'users' | 'list' | 'users_blacklist';
 | 
					          src?: 'home' | 'all' | 'users' | 'list' | 'users_blacklist';
 | 
				
			||||||
          /** Format: misskey:id */
 | 
					          /** Format: misskey:id */
 | 
				
			||||||
          userListId?: string | null;
 | 
					          userListId?: string | null;
 | 
				
			||||||
          keywords: string[][];
 | 
					          keywords?: string[][];
 | 
				
			||||||
          excludeKeywords: string[][];
 | 
					          excludeKeywords?: string[][];
 | 
				
			||||||
          users: string[];
 | 
					          users?: string[];
 | 
				
			||||||
          caseSensitive: boolean;
 | 
					          caseSensitive?: boolean;
 | 
				
			||||||
          localOnly?: boolean;
 | 
					          localOnly?: boolean;
 | 
				
			||||||
          withReplies: boolean;
 | 
					          withReplies?: boolean;
 | 
				
			||||||
          withFile: boolean;
 | 
					          withFile?: boolean;
 | 
				
			||||||
          notify: boolean;
 | 
					          notify?: boolean;
 | 
				
			||||||
        };
 | 
					        };
 | 
				
			||||||
      };
 | 
					      };
 | 
				
			||||||
    };
 | 
					    };
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,3 +1,8 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SPDX-FileCopyrightText: syuilo and misskey-project
 | 
				
			||||||
 | 
					 * SPDX-License-Identifier: AGPL-3.0-only
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import { createWriteStream } from 'node:fs';
 | 
					import { createWriteStream } from 'node:fs';
 | 
				
			||||||
import { mkdir } from 'node:fs/promises';
 | 
					import { mkdir } from 'node:fs/promises';
 | 
				
			||||||
import { resolve } from 'node:path';
 | 
					import { resolve } from 'node:path';
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue