Compare commits
	
		
			215 Commits
		
	
	
		
			13.0.0-bet
			...
			13.1.0-bet
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 951ab90b1a | ||
|   | 7e89e70934 | ||
|   | 0b4a7e8166 | ||
|   | 59748f07d1 | ||
|   | 65cd605b73 | ||
|   | b8afabde2c | ||
|   | 02b6595d76 | ||
|   | 336d8fe785 | ||
|   | ed9a49687e | ||
|   | a160b01cff | ||
|   | d05ffc0a7c | ||
|   | afc0be6790 | ||
|   | 85f3df4c0e | ||
|   | eea47ca2e8 | ||
|   | 57b1fe44d4 | ||
|   | 79212bbd37 | ||
|   | d456308653 | ||
|   | 424919ffd0 | ||
|   | d75225e23b | ||
|   | 8f0c598772 | ||
|   | fe4fbafcf0 | ||
|   | 0db2abd56c | ||
|   | c62a4d6282 | ||
|   | 0de41063da | ||
|   | d79478c265 | ||
|   | f8d0902080 | ||
|   | a69c78e709 | ||
|   | 84b8ffb7d0 | ||
|   | 3feaf39294 | ||
|   | fe98ad8849 | ||
|   | 65577e43c8 | ||
|   | 9d64ac6d6f | ||
|   | e13434c2f0 | ||
|   | 5416a295c1 | ||
|   | 119c650406 | ||
|   | 57386f46d2 | ||
|   | 77e491f52c | ||
|   | 6f1243f722 | ||
|   | fe0bb21b37 | ||
|   | 60d9bb0218 | ||
|   | 956375e2e8 | ||
|   | 0fa602a184 | ||
|   | d600296360 | ||
|   | c9f5e60f43 | ||
|   | d513848f65 | ||
|   | ae6af6aefd | ||
|   | a0ae9f7593 | ||
|   | dace5b6940 | ||
|   | 2d8b97287e | ||
|   | ec63a50de2 | ||
|   | 6e2d7e9792 | ||
|   | 39349dcba5 | ||
|   | a5b1fe5d16 | ||
|   | 91bbb67e4a | ||
|   | f368bce9d5 | ||
|   | fbfe42d6f0 | ||
|   | f3c5ca6cf4 | ||
|   | 0022267072 | ||
|   | 30fced38c4 | ||
|   | 7e5f3dbf11 | ||
|   | 9f0dfb5517 | ||
|   | 678c7d9502 | ||
|   | 91a3c3943d | ||
|   | c46b45a467 | ||
|   | 9385767b12 | ||
|   | 7795ff0c95 | ||
|   | a9acd72eb7 | ||
|   | 67d366c3ca | ||
|   | 1f8f051ee2 | ||
|   | 94004b7a3f | ||
|   | 3e9f88506e | ||
|   | 81f11d8f86 | ||
|   | 518b3e2f73 | ||
|   | d0157b3bfd | ||
|   | 7fc8d2e6d5 | ||
|   | fb0f9711ba | ||
|   | 92136272b0 | ||
|   | e1159e9ef2 | ||
|   | a2e61c6708 | ||
|   | 726959911c | ||
|   | d59914b959 | ||
|   | 07025caee9 | ||
|   | 1c0289e490 | ||
|   | 275fcd8bbc | ||
|   | 0c0aa93668 | ||
|   | bfcd5ea440 | ||
|   | 3ff43cca02 | ||
|   | 6bd536c526 | ||
|   | 7738a36014 | ||
|   | daddec8362 | ||
|   | a3832d73fd | ||
|   | cedb4267ba | ||
|   | 9c6629d582 | ||
|   | 4ee4e70ee0 | ||
|   | bb7867351c | ||
|   | fea7460930 | ||
|   | 1bf2bf1773 | ||
|   | 3d668ad10d | ||
|   | 2801338a3c | ||
|   | b66f4ebba1 | ||
|   | 9ee1b5f30a | ||
|   | 0f31a0548c | ||
|   | ffc29aa6f5 | ||
|   | d23aa94b41 | ||
|   | c1b6378951 | ||
|   | bb5d2bda51 | ||
|   | d075471b2d | ||
|   | 199d98bf79 | ||
|   | 3ae798d526 | ||
|   | e1bd61c70e | ||
|   | 0296f841c3 | ||
|   | bd1f4b8d98 | ||
|   | dc19f20153 | ||
|   | f5cd809f62 | ||
|   | 09d5a7806a | ||
|   | 4606f23ed8 | ||
|   | 8451e08aaa | ||
|   | 2047449294 | ||
|   | d61eee695f | ||
|   | 73b62797cd | ||
|   | 170cfc6a0e | ||
|   | 6bf1d7e398 | ||
|   | e46e7f5252 | ||
|   | 5952f1ac24 | ||
|   | a08369fe36 | ||
|   | 6cb9612943 | ||
|   | 76c049522e | ||
|   | c41879c542 | ||
|   | 99bdb11d24 | ||
|   | c2009acb2d | ||
|   | 46d2a8726e | ||
|   | 7df3ca7388 | ||
|   | 51b8d4ae3e | ||
|   | ab1124abba | ||
|   | 3db84a2e8f | ||
|   | 9a78bbf0f1 | ||
|   | efbec444e8 | ||
|   | 2f06f2a6da | ||
|   | b8da51e08c | ||
|   | af6a578fa6 | ||
|   | 73d735a1f7 | ||
|   | b8b1899a9f | ||
|   | d52f0617a1 | ||
|   | c730973294 | ||
|   | 2c2e064871 | ||
|   | e3c39d4b52 | ||
|   | 5da74897ae | ||
|   | 4b1009b34e | ||
|   | 203a7ad073 | ||
|   | 34a7b52105 | ||
|   | 30fc166c08 | ||
|   | c84d86b368 | ||
|   | 1e5d4db0a1 | ||
|   | 5e02f0d325 | ||
|   | ce5506f331 | ||
|   | 91105845d8 | ||
|   | 2bedc084a3 | ||
|   | 027ef1ea4a | ||
|   | 668aa17eef | ||
|   | ebf8ef22e4 | ||
|   | bcb5182e86 | ||
|   | f45059b7b1 | ||
|   | d0aee58599 | ||
|   | 68e65ed5df | ||
|   | 367ccb9971 | ||
|   | 4151087d3c | ||
|   | 39c058a4bb | ||
|   | d1807ee5dc | ||
|   | e6a76b31be | ||
|   | 98469117bf | ||
|   | a5becfc042 | ||
|   | d2204fd5c8 | ||
|   | 519a08f8b5 | ||
|   | 303519a1bd | ||
|   | 161da24841 | ||
|   | 6e40024660 | ||
|   | 73c78d4c38 | ||
|   | 2654936c17 | ||
|   | 23810e3e1e | ||
|   | d6c89bf003 | ||
|   | 49ab2a5f93 | ||
|   | bc0b8afb1f | ||
|   | b250456814 | ||
|   | 0a6e237d09 | ||
|   | 54ff4e53cb | ||
|   | 002ccbb5f0 | ||
|   | 7b7faf1e84 | ||
|   | 9936088200 | ||
|   | 990f4b52bd | ||
|   | 4c21d83639 | ||
|   | d43a4a2d46 | ||
|   | 8d2c3bb18d | ||
|   | 4e39e690b6 | ||
|   | 6458239a7c | ||
|   | a5aaa032ca | ||
|   | 71bbef69c7 | ||
|   | c5c40a73b7 | ||
|   | 74910f8d70 | ||
|   | e00003edff | ||
|   | bedb98185e | ||
|   | da6f955d58 | ||
|   | 6bdccea26b | ||
|   | b2117ba3a1 | ||
|   | ba349fc62f | ||
|   | b2c79a5f2c | ||
|   | 3e415e733d | ||
|   | a5e84e5de9 | ||
|   | 8673353029 | ||
|   | 4579d02296 | ||
|   | 978a9bbb3b | ||
|   | 2470afaa2e | ||
|   | 60e545b2fd | ||
|   | 6555644b88 | ||
|   | df56bd6d57 | ||
|   | e51432a461 | 
							
								
								
									
										151
									
								
								.config/docker_example.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										151
									
								
								.config/docker_example.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,151 @@ | ||||
| #━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ | ||||
| # Misskey configuration | ||||
| #━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ | ||||
|  | ||||
| #   ┌─────┐ | ||||
| #───┘ URL └───────────────────────────────────────────────────── | ||||
|  | ||||
| # Final accessible URL seen by a user. | ||||
| url: https://example.tld/ | ||||
|  | ||||
| # ONCE YOU HAVE STARTED THE INSTANCE, DO NOT CHANGE THE | ||||
| # URL SETTINGS AFTER THAT! | ||||
|  | ||||
| #   ┌───────────────────────┐ | ||||
| #───┘ Port and TLS settings └─────────────────────────────────── | ||||
|  | ||||
| # | ||||
| # Misskey requires a reverse proxy to support HTTPS connections. | ||||
| # | ||||
| #                 +----- https://example.tld/ ------------+ | ||||
| #   +------+      |+-------------+      +----------------+| | ||||
| #   | User | ---> || Proxy (443) | ---> | Misskey (3000) || | ||||
| #   +------+      |+-------------+      +----------------+| | ||||
| #                 +---------------------------------------+ | ||||
| # | ||||
| #   You need to set up a reverse proxy. (e.g. nginx) | ||||
| #   An encrypted connection with HTTPS is highly recommended | ||||
| #   because tokens may be transferred in GET requests. | ||||
|  | ||||
| # The port that your Misskey server should listen on. | ||||
| port: 3000 | ||||
|  | ||||
| #   ┌──────────────────────────┐ | ||||
| #───┘ PostgreSQL configuration └──────────────────────────────── | ||||
|  | ||||
| db: | ||||
|   host: db | ||||
|   port: 5432 | ||||
|  | ||||
|   # Database name | ||||
|   db: misskey | ||||
|  | ||||
|   # Auth | ||||
|   user: example-misskey-user | ||||
|   pass: example-misskey-pass | ||||
|  | ||||
|   # Whether disable Caching queries | ||||
|   #disableCache: true | ||||
|  | ||||
|   # Extra Connection options | ||||
|   #extra: | ||||
|   #  ssl: true | ||||
|  | ||||
| #   ┌─────────────────────┐ | ||||
| #───┘ Redis configuration └───────────────────────────────────── | ||||
|  | ||||
| redis: | ||||
|   host: redis | ||||
|   port: 6379 | ||||
|   #family: 0  # 0=Both, 4=IPv4, 6=IPv6 | ||||
|   #pass: example-pass | ||||
|   #prefix: example-prefix | ||||
|   #db: 1 | ||||
|  | ||||
| #   ┌─────────────────────────────┐ | ||||
| #───┘ Elasticsearch configuration └───────────────────────────── | ||||
|  | ||||
| #elasticsearch: | ||||
| #  host: localhost | ||||
| #  port: 9200 | ||||
| #  ssl: false | ||||
| #  user: | ||||
| #  pass: | ||||
|  | ||||
| #   ┌───────────────┐ | ||||
| #───┘ ID generation └─────────────────────────────────────────── | ||||
|  | ||||
| # You can select the ID generation method. | ||||
| # You don't usually need to change this setting, but you can | ||||
| # change it according to your preferences. | ||||
|  | ||||
| # Available methods: | ||||
| # aid ... Short, Millisecond accuracy | ||||
| # meid ... Similar to ObjectID, Millisecond accuracy | ||||
| # ulid ... Millisecond accuracy | ||||
| # objectid ... This is left for backward compatibility | ||||
|  | ||||
| # ONCE YOU HAVE STARTED THE INSTANCE, DO NOT CHANGE THE | ||||
| # ID SETTINGS AFTER THAT! | ||||
|  | ||||
| id: 'aid' | ||||
|  | ||||
| #   ┌─────────────────────┐ | ||||
| #───┘ Other configuration └───────────────────────────────────── | ||||
|  | ||||
| # Whether disable HSTS | ||||
| #disableHsts: true | ||||
|  | ||||
| # Number of worker processes | ||||
| #clusterLimit: 1 | ||||
|  | ||||
| # Job concurrency per worker | ||||
| # deliverJobConcurrency: 128 | ||||
| # inboxJobConcurrency: 16 | ||||
|  | ||||
| # Job rate limiter | ||||
| # deliverJobPerSec: 128 | ||||
| # inboxJobPerSec: 16 | ||||
|  | ||||
| # Job attempts | ||||
| # deliverJobMaxAttempts: 12 | ||||
| # inboxJobMaxAttempts: 8 | ||||
|  | ||||
| # IP address family used for outgoing request (ipv4, ipv6 or dual) | ||||
| #outgoingAddressFamily: ipv4 | ||||
|  | ||||
| # Syslog option | ||||
| #syslog: | ||||
| #  host: localhost | ||||
| #  port: 514 | ||||
|  | ||||
| # Proxy for HTTP/HTTPS | ||||
| #proxy: http://127.0.0.1:3128 | ||||
|  | ||||
| proxyBypassHosts: | ||||
|   - api.deepl.com | ||||
|   - api-free.deepl.com | ||||
|   - www.recaptcha.net | ||||
|   - hcaptcha.com | ||||
|   - challenges.cloudflare.com | ||||
|  | ||||
| # Proxy for SMTP/SMTPS | ||||
| #proxySmtp: http://127.0.0.1:3128   # use HTTP/1.1 CONNECT | ||||
| #proxySmtp: socks4://127.0.0.1:1080 # use SOCKS4 | ||||
| #proxySmtp: socks5://127.0.0.1:1080 # use SOCKS5 | ||||
|  | ||||
| # Media Proxy | ||||
| #mediaProxy: https://example.com/proxy | ||||
|  | ||||
| # Proxy remote files (default: false) | ||||
| #proxyRemoteFiles: true | ||||
|  | ||||
| # Sign to ActivityPub GET request (default: true) | ||||
| signToActivityPubGet: true | ||||
|  | ||||
| #allowedPrivateNetworks: [ | ||||
| #  '127.0.0.1/32' | ||||
| #] | ||||
|  | ||||
| # Upload or download file size limits (bytes) | ||||
| #maxFileSize: 262144000 | ||||
| @@ -122,10 +122,12 @@ id: 'aid' | ||||
| # Proxy for HTTP/HTTPS | ||||
| #proxy: http://127.0.0.1:3128 | ||||
|  | ||||
| #proxyBypassHosts: [ | ||||
| #  'example.com', | ||||
| #  '192.0.2.8' | ||||
| #] | ||||
| proxyBypassHosts: | ||||
|   - api.deepl.com | ||||
|   - api-free.deepl.com | ||||
|   - www.recaptcha.net | ||||
|   - hcaptcha.com | ||||
|   - challenges.cloudflare.com | ||||
|  | ||||
| # Proxy for SMTP/SMTPS | ||||
| #proxySmtp: http://127.0.0.1:3128   # use HTTP/1.1 CONNECT | ||||
|   | ||||
							
								
								
									
										10
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										10
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
								
							| @@ -5,6 +5,11 @@ | ||||
|  | ||||
| version: 2 | ||||
| updates: | ||||
| - package-ecosystem: github-actions | ||||
|   directory: "/" | ||||
|   schedule: | ||||
|     interval: daily | ||||
|   open-pull-requests-limit: 0 | ||||
| - package-ecosystem: npm | ||||
|   directory: "/" | ||||
|   schedule: | ||||
| @@ -20,3 +25,8 @@ updates: | ||||
|   schedule: | ||||
|     interval: daily | ||||
|   open-pull-requests-limit: 0 | ||||
| - package-ecosystem: npm | ||||
|   directory: "/packages/sw" | ||||
|   schedule: | ||||
|     interval: daily | ||||
|   open-pull-requests-limit: 0 | ||||
|   | ||||
							
								
								
									
										18
									
								
								.github/workflows/check_copyright_year.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								.github/workflows/check_copyright_year.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| name: Check copyright year | ||||
|  | ||||
| on: | ||||
|   push: | ||||
|     branches: | ||||
|       - master | ||||
|       - develop | ||||
|  | ||||
| jobs: | ||||
|   check_copyright_year: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|     - uses: actions/checkout@v3.2.0 | ||||
|     - run: | | ||||
|         if [ "$(grep Copyright COPYING | sed -e 's/.*2014-\([0-9]*\) .*/\1/g')" -ne "$(date +%Y)" ]; then | ||||
|           echo "Please change copyright year!" | ||||
|           exit 1 | ||||
|         fi | ||||
							
								
								
									
										10
									
								
								.github/workflows/docker-develop.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										10
									
								
								.github/workflows/docker-develop.yml
									
									
									
									
										vendored
									
									
								
							| @@ -10,22 +10,22 @@ jobs: | ||||
|   push_to_registry: | ||||
|     name: Push Docker image to Docker Hub | ||||
|     runs-on: ubuntu-latest | ||||
|  | ||||
|     if: github.repository == 'misskey-dev/misskey' | ||||
|     steps: | ||||
|       - name: Check out the repo | ||||
|         uses: actions/checkout@v2 | ||||
|         uses: actions/checkout@v3.3.0 | ||||
|       - name: Docker meta | ||||
|         id: meta | ||||
|         uses: docker/metadata-action@v3 | ||||
|         uses: docker/metadata-action@v4 | ||||
|         with: | ||||
|           images: misskey/misskey | ||||
|       - name: Log in to Docker Hub | ||||
|         uses: docker/login-action@v1 | ||||
|         uses: docker/login-action@v2 | ||||
|         with: | ||||
|           username: ${{ secrets.DOCKER_USERNAME }} | ||||
|           password: ${{ secrets.DOCKER_PASSWORD }} | ||||
|       - name: Build and Push to Docker Hub | ||||
|         uses: docker/build-push-action@v2 | ||||
|         uses: docker/build-push-action@v3 | ||||
|         with: | ||||
|           context: . | ||||
|           push: true | ||||
|   | ||||
							
								
								
									
										8
									
								
								.github/workflows/docker.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										8
									
								
								.github/workflows/docker.yml
									
									
									
									
										vendored
									
									
								
							| @@ -12,10 +12,10 @@ jobs: | ||||
|  | ||||
|     steps: | ||||
|       - name: Check out the repo | ||||
|         uses: actions/checkout@v2 | ||||
|         uses: actions/checkout@v3.3.0 | ||||
|       - name: Docker meta | ||||
|         id: meta | ||||
|         uses: docker/metadata-action@v3 | ||||
|         uses: docker/metadata-action@v4 | ||||
|         with: | ||||
|           images: misskey/misskey | ||||
|           tags: | | ||||
| @@ -26,12 +26,12 @@ jobs: | ||||
|             type=semver,pattern={{major}}.{{minor}} | ||||
|             type=semver,pattern={{major}} | ||||
|       - name: Log in to Docker Hub | ||||
|         uses: docker/login-action@v1 | ||||
|         uses: docker/login-action@v2 | ||||
|         with: | ||||
|           username: ${{ secrets.DOCKER_USERNAME }} | ||||
|           password: ${{ secrets.DOCKER_PASSWORD }} | ||||
|       - name: Build and Push to Docker Hub | ||||
|         uses: docker/build-push-action@v2 | ||||
|         uses: docker/build-push-action@v3 | ||||
|         with: | ||||
|           context: . | ||||
|           push: true | ||||
|   | ||||
							
								
								
									
										30
									
								
								.github/workflows/lint.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										30
									
								
								.github/workflows/lint.yml
									
									
									
									
										vendored
									
									
								
							| @@ -8,22 +8,26 @@ on: | ||||
|   pull_request: | ||||
|  | ||||
| jobs: | ||||
|   yarn_install: | ||||
|   pnpm_install: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|     - uses: actions/checkout@v2 | ||||
|     - uses: actions/checkout@v3.3.0 | ||||
|       with: | ||||
|         fetch-depth: 0 | ||||
|         submodules: true | ||||
|     - uses: actions/setup-node@v3.2.0 | ||||
|     - uses: pnpm/action-setup@v2 | ||||
|       with: | ||||
|         version: 7 | ||||
|         run_install: false | ||||
|     - uses: actions/setup-node@v3.6.0 | ||||
|       with: | ||||
|         node-version: 18.x | ||||
|         cache: 'yarn' | ||||
|         cache: 'pnpm' | ||||
|     - run: corepack enable | ||||
|     - run: yarn install --immutable | ||||
|     - run: pnpm i --frozen-lockfile | ||||
|  | ||||
|   lint: | ||||
|     needs: [yarn_install] | ||||
|     needs: [pnpm_install] | ||||
|     runs-on: ubuntu-latest | ||||
|     continue-on-error: true | ||||
|     strategy: | ||||
| @@ -33,14 +37,18 @@ jobs: | ||||
|         - frontend | ||||
|         - sw | ||||
|     steps: | ||||
|     - uses: actions/checkout@v2 | ||||
|     - uses: actions/checkout@v3.3.0 | ||||
|       with: | ||||
|         fetch-depth: 0 | ||||
|         submodules: true | ||||
|     - uses: actions/setup-node@v3.2.0 | ||||
|     - uses: pnpm/action-setup@v2 | ||||
|       with: | ||||
|         version: 7 | ||||
|         run_install: false | ||||
|     - uses: actions/setup-node@v3.6.0 | ||||
|       with: | ||||
|         node-version: 18.x | ||||
|         cache: 'yarn' | ||||
|         cache: 'pnpm' | ||||
|     - run: corepack enable | ||||
|     - run: yarn install --immutable | ||||
|     - run: yarn workspace ${{ matrix.workspace }} run lint | ||||
|     - run: pnpm i --frozen-lockfile | ||||
|     - run: pnpm --filter ${{ matrix.workspace }} run lint | ||||
|   | ||||
							
								
								
									
										11
									
								
								.github/workflows/pr-preview-deploy.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										11
									
								
								.github/workflows/pr-preview-deploy.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,7 +1,5 @@ | ||||
| # Run secret-dependent integration tests only after /deploy approval | ||||
| on: | ||||
|   pull_request: | ||||
|     types: [opened, reopened, synchronize] | ||||
|   repository_dispatch: | ||||
|     types: [deploy-command] | ||||
|  | ||||
| @@ -12,11 +10,10 @@ jobs: | ||||
|   deploy-preview-environment: | ||||
|     runs-on: ubuntu-latest | ||||
|     if: | ||||
|       github.event_name == 'repository_dispatch' && | ||||
|       github.event.client_payload.slash_command.sha != '' && | ||||
|       contains(github.event.client_payload.pull_request.head.sha, github.event.client_payload.slash_command.sha) | ||||
|     steps: | ||||
|     - uses: actions/github-script@v5 | ||||
|     - uses: actions/github-script@v6.3.3 | ||||
|       id: check-id | ||||
|       env: | ||||
|         number: ${{ github.event.client_payload.pull_request.number }} | ||||
| @@ -40,7 +37,7 @@ jobs: | ||||
|  | ||||
|           return check[0].id; | ||||
|  | ||||
|     - uses: actions/github-script@v5 | ||||
|     - uses: actions/github-script@v6.3.3 | ||||
|       env: | ||||
|         check_id: ${{ steps.check-id.outputs.result }} | ||||
|         details_url: ${{ github.server_url }}/${{ github.repository }}/runs/${{ github.run_id }} | ||||
| @@ -56,7 +53,7 @@ jobs: | ||||
|  | ||||
|     # Check out merge commit | ||||
|     - name: Fork based /deploy checkout | ||||
|       uses: actions/checkout@v2 | ||||
|       uses: actions/checkout@v3.3.0 | ||||
|       with: | ||||
|         ref: 'refs/pull/${{ github.event.client_payload.pull_request.number }}/merge' | ||||
|  | ||||
| @@ -75,7 +72,7 @@ jobs: | ||||
|         timeout: 15m | ||||
|  | ||||
|     # Update check run called "integration-fork" | ||||
|     - uses: actions/github-script@v5 | ||||
|     - uses: actions/github-script@v6.3.3 | ||||
|       id: update-check-run | ||||
|       if: ${{ always() }} | ||||
|       env: | ||||
|   | ||||
							
								
								
									
										1
									
								
								.github/workflows/pr-preview-destroy.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.github/workflows/pr-preview-destroy.yml
									
									
									
									
										vendored
									
									
								
							| @@ -9,6 +9,7 @@ name: Destroy preview environment | ||||
| jobs: | ||||
|   destroy-preview-environment: | ||||
|     runs-on: ubuntu-latest | ||||
|     if: github.repository == github.event.pull_request.head.repo.full_name | ||||
|     steps: | ||||
|       - name: Context | ||||
|         uses: okteto/context@latest | ||||
|   | ||||
							
								
								
									
										41
									
								
								.github/workflows/test.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										41
									
								
								.github/workflows/test.yml
									
									
									
									
										vendored
									
									
								
							| @@ -23,31 +23,35 @@ jobs: | ||||
|         env: | ||||
|           POSTGRES_DB: test-misskey | ||||
|           POSTGRES_HOST_AUTH_METHOD: trust | ||||
|           YARN_CHECKSUM_BEHAVIOR: update | ||||
|       redis: | ||||
|         image: redis:6 | ||||
|         ports: | ||||
|           - 56312:6379 | ||||
|  | ||||
|     steps: | ||||
|     - uses: actions/checkout@v2 | ||||
|     - uses: actions/checkout@v3.3.0 | ||||
|       with: | ||||
|         submodules: true | ||||
|     - name: Install pnpm | ||||
|       uses: pnpm/action-setup@v2 | ||||
|       with: | ||||
|         version: 7 | ||||
|         run_install: false | ||||
|     - name: Use Node.js ${{ matrix.node-version }} | ||||
|       uses: actions/setup-node@v3.2.0 | ||||
|       uses: actions/setup-node@v3.6.0 | ||||
|       with: | ||||
|         node-version: ${{ matrix.node-version }} | ||||
|         cache: 'yarn' | ||||
|         cache: 'pnpm' | ||||
|     - run: corepack enable | ||||
|     - run: yarn install --immutable | ||||
|     - name: Check yarn.lock | ||||
|       run: git diff --exit-code yarn.lock | ||||
|     - run: pnpm i --frozen-lockfile | ||||
|     - name: Check pnpm-lock.yaml | ||||
|       run: git diff --exit-code pnpm-lock.yaml | ||||
|     - name: Copy Configure | ||||
|       run: cp .github/misskey/test.yml .config | ||||
|     - name: Build | ||||
|       run: yarn build | ||||
|       run: pnpm build | ||||
|     - name: Test | ||||
|       run: yarn jest-and-coverage | ||||
|       run: pnpm jest-and-coverage | ||||
|     - name: Upload Coverage | ||||
|       uses: codecov/codecov-action@v3 | ||||
|       with: | ||||
| @@ -77,7 +81,7 @@ jobs: | ||||
|           - 56312:6379 | ||||
|  | ||||
|     steps: | ||||
|     - uses: actions/checkout@v2 | ||||
|     - uses: actions/checkout@v3.3.0 | ||||
|       with: | ||||
|         submodules: true | ||||
|     # https://github.com/cypress-io/cypress-docker-images/issues/150 | ||||
| @@ -86,19 +90,22 @@ jobs: | ||||
|     #  if: ${{ matrix.browser == 'firefox' }} | ||||
|     #- uses: browser-actions/setup-firefox@latest | ||||
|     #  if: ${{ matrix.browser == 'firefox' }} | ||||
|     - name: Install pnpm | ||||
|       uses: pnpm/action-setup@v2 | ||||
|       with: | ||||
|         version: 7 | ||||
|         run_install: false | ||||
|     - name: Use Node.js ${{ matrix.node-version }} | ||||
|       uses: actions/setup-node@v3.2.0 | ||||
|       uses: actions/setup-node@v3.6.0 | ||||
|       with: | ||||
|         node-version: ${{ matrix.node-version }} | ||||
|         cache: 'yarn' | ||||
|         cache: 'pnpm' | ||||
|     - run: corepack enable | ||||
|     - run: yarn install --immutable | ||||
|       env: | ||||
|         YARN_CHECKSUM_BEHAVIOR: update | ||||
|     - run: pnpm i --frozen-lockfile | ||||
|     - name: Copy Configure | ||||
|       run: cp .github/misskey/test.yml .config | ||||
|     - name: Build | ||||
|       run: yarn build | ||||
|       run: pnpm build | ||||
|     # https://github.com/cypress-io/cypress/issues/4351#issuecomment-559489091 | ||||
|     - name: ALSA Env | ||||
|       run: echo -e 'pcm.!default {\n type hw\n card 0\n}\n\nctl.!default {\n type hw\n card 0\n}' > ~/.asoundrc | ||||
| @@ -106,7 +113,7 @@ jobs: | ||||
|       uses: cypress-io/github-action@v4 | ||||
|       with: | ||||
|         install: false | ||||
|         start: yarn start:test | ||||
|         start: pnpm start:test | ||||
|         wait-on: 'http://localhost:61812' | ||||
|         headless: false | ||||
|         browser: ${{ matrix.browser }} | ||||
|   | ||||
							
								
								
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -30,6 +30,7 @@ coverage | ||||
| # config | ||||
| /.config/* | ||||
| !/.config/example.yml | ||||
| !/.config/docker_example.yml | ||||
| !/.config/docker_example.env | ||||
|  | ||||
| # misskey | ||||
|   | ||||
							
								
								
									
										5
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | ||||
| { | ||||
|     "search.exclude": { | ||||
|         "**/node_modules": true | ||||
|     } | ||||
| } | ||||
							
								
								
									
										546
									
								
								.yarn/plugins/@yarnpkg/plugin-interactive-tools.cjs
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										546
									
								
								.yarn/plugins/@yarnpkg/plugin-interactive-tools.cjs
									
									
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										807
									
								
								.yarn/releases/yarn-3.3.0.cjs
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										807
									
								
								.yarn/releases/yarn-3.3.0.cjs
									
									
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										42
									
								
								.yarnrc.yml
									
									
									
									
									
								
							
							
						
						
									
										42
									
								
								.yarnrc.yml
									
									
									
									
									
								
							| @@ -1,42 +0,0 @@ | ||||
| httpTimeout: 600000 | ||||
|  | ||||
| nmHoistingLimits: none | ||||
|  | ||||
| nodeLinker: pnpm | ||||
|  | ||||
| packageExtensions: | ||||
|   "@bull-board/api@*": | ||||
|     peerDependencies: | ||||
|       "@bull-board/ui": "*" | ||||
|   "@tensorflow/tfjs@*": | ||||
|     dependencies: | ||||
|       long: "*" | ||||
|   chartjs-adapter-date-fns@*: | ||||
|     peerDependencies: | ||||
|       date-fns: "*" | ||||
|   consolidate@*: | ||||
|     dependencies: | ||||
|       ejs: "*" | ||||
|   # these are needed to extend fastify types | ||||
|   "@fastify/accepts@*": | ||||
|     peerDependencies: | ||||
|       fastify: "*" | ||||
|   "@fastify/cookie@*": | ||||
|     peerDependencies: | ||||
|       fastify: "*" | ||||
|   "@fastify/static@*": | ||||
|     peerDependencies: | ||||
|       fastify: "*" | ||||
|   "@fastify/view@*": | ||||
|     peerDependencies: | ||||
|       fastify: "*" | ||||
|  | ||||
| plugins: | ||||
|   - path: .yarn/plugins/@yarnpkg/plugin-interactive-tools.cjs | ||||
|     spec: "@yarnpkg/plugin-interactive-tools" | ||||
|   - path: .yarn/plugins/@yarnpkg/plugin-workspace-tools.cjs | ||||
|     spec: "@yarnpkg/plugin-workspace-tools" | ||||
|  | ||||
| progressBarStyle: patrick | ||||
|  | ||||
| yarnPath: .yarn/releases/yarn-3.3.0.cjs | ||||
							
								
								
									
										68
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										68
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -9,15 +9,36 @@ | ||||
| You should also include the user name that made the change. | ||||
| --> | ||||
|  | ||||
| ## 13.0.0 (unreleased) | ||||
| ## 13.x.x (unreleased) | ||||
|  | ||||
| ### Improvements | ||||
| - 実績機能 | ||||
| - Playのプリセットを追加 | ||||
| - Playのscriptの文字数制限を緩和 | ||||
| - AiScript GUIの強化 | ||||
| - リアクション一覧詳細ダイアログを表示できるように | ||||
| - 存在しないカスタム絵文字をテキストで表示するように | ||||
| - Alt text in image viewer | ||||
| - ジョブキューのプロセスとWebサーバーのプロセスを分離 | ||||
|  | ||||
| ### Bugfixes | ||||
| - playを削除する手段がなかったのを修正 | ||||
|  | ||||
| ## 13.0.0 (2023/01/16) | ||||
|  | ||||
| ### TL;DR | ||||
| - New features (Play, new widgets, new charts, 🍪👈, etc) | ||||
| - New features (Role system, Misskey Play, New widgets, New charts, 🍪👈, etc) | ||||
| - Rewriten backend | ||||
| - Better performance (backend and frontend) | ||||
| - Various usability improvements | ||||
| - Various UI tweaks | ||||
|  | ||||
| ### Notable features | ||||
| - ロール機能 | ||||
| 	- 従来より柔軟にユーザーのポリシーを管理できます。例えば、「インスタンスのパトロンはアンテナを30個まで作れる」「基本的にLTLは見れないが、許可した人だけ見れる」「招待制インスタンスだけどユーザーなら誰でも他者を招待できる」のような運用はもちろん、「ローカルユーザーかつアカウント作成から1日未満のユーザーはパブリックな投稿を行えない」のように複数条件を組み合わせて、自動でロールを付与する設定も可能です。 | ||||
| - Misskey Play | ||||
| 	- 従来の動的なPagesに代わる、新しいプラットフォームです。動的なコンテンツ(アプリケーション)に特化していて、Pagesに比べてはるかに柔軟なアプリケーションを作成可能です。 | ||||
|  | ||||
| ### Changes | ||||
| #### For server admins | ||||
| - Node.js 18.x or later is required | ||||
| @@ -25,19 +46,30 @@ You should also include the user name that made the change. | ||||
| 	- Misskey not using 15 specific features at 13.0.0, but may do so in the future. | ||||
| - Elasticsearchのサポートが削除されました | ||||
| 	- 代わりに今後任意の検索プロバイダを設定できる仕組みを構想しています。その仕組みを使えば今まで通りElasticsearchも利用できます | ||||
| - Migrate to Yarn Berry (v3.2.1) @ThatOneCalculator | ||||
| 	- You may have to `yarn run clean-all`, `sudo corepack enable` and `yarn set version berry` before running `yarn install` if you're still on yarn classic | ||||
| - Yarnからpnpmに移行されました | ||||
|   corepackの有効化を推奨します: `sudo corepack enable` | ||||
| - インスタンスブロックはサブドメインにも適用されるようになります | ||||
| - ロールの導入に伴い、いくつかの機能がロールと統合されました | ||||
| 	- モデレーターはロールに統合されました。今までのモデレーター情報は失われるため、予めモデレーター一覧を記録しておき、アップデート後にモデレーターロールを作りアサインし直してください。 | ||||
| 	- サイレンスはロールに統合されました。今までのユーザーは恩赦されるため、予めサイレンス一覧を記録しておくのをおすすめします。 | ||||
| 	- ユーザーごとのドライブ容量設定はロールに統合されました。 | ||||
| 	- インスタンスデフォルトのドライブ容量設定はロールに統合されました。アップデート後、ベースロールもしくはコンディショナルロールでドライブ容量を編集してください。 | ||||
| 	- LTL/GTLの解放状態はロールに統合されました。 | ||||
| - Dockerの実行をrootで行わないようにしました。Dockerかつオブジェクトストレージを使用していない場合は`chown -hR 991.991 ./files`を実行してください。   | ||||
|   https://github.com/misskey-dev/misskey/pull/9560 | ||||
|  | ||||
| #### For users | ||||
| - ノートのウォッチ機能が削除されました | ||||
| - アンケートに投票された際に通知が作成されなくなりました | ||||
| - ノートの数式埋め込みが削除されました | ||||
| - 新たに動的なPagesを作ることはできなくなりました | ||||
| 	- 代わりにAiScriptを用いてより柔軟に動的なコンテンツを作成できるMisskey Play機能が実装されています。 | ||||
| - AiScriptが0.12.2にアップデートされました | ||||
| 	- 0.12.xの変更点についてはこちら https://github.com/syuilo/aiscript/blob/master/CHANGELOG.md#0120 | ||||
| 	- 0.12.x未満のプラグインは読み込むことはできません | ||||
| - iOS15以下のデバイスはサポートされなくなりました | ||||
| - Firefox109以下はサポートされなくなりました | ||||
| - Firefox110以下はサポートされなくなりました | ||||
|   - 109でもContainerQueriesのフラグを有効にする事で問題なく使用できます | ||||
|  | ||||
| #### For app developers | ||||
| - API: metaのレスポンスに`emojis`プロパティが含まれなくなりました | ||||
| @@ -49,8 +81,10 @@ You should also include the user name that made the change. | ||||
| - API: `user`および`note`エンティティに`emojis`プロパティが含まれなくなりました | ||||
| - API: `user`エンティティに`avatarColor`および`bannerColor`プロパティが含まれなくなりました | ||||
| - API: `instance`エンティティに`latestStatus`、`lastCommunicatedAt`、`latestRequestSentAt`プロパティが含まれなくなりました | ||||
| - API: `instance`エンティティの`caughtAt`は`firstRetrievedAt`に名前が変わりました | ||||
|  | ||||
| ### Improvements | ||||
| - Role system @syuilo | ||||
| - Misskey Play @syuilo | ||||
| - Introduce retention-rate aggregation @syuilo | ||||
| - Make possible to export favorited notes @syuilo | ||||
| @@ -58,10 +92,22 @@ You should also include the user name that made the change. | ||||
| - Push notification of Antenna note @tamaina | ||||
| - AVIF support @tamaina | ||||
| - Add Cloudflare Turnstile CAPTCHA support @CyberRex0 | ||||
| - レートリミットをユーザーごとに調整可能に @syuilo | ||||
| - 非モデレーターでも、権限を持つロールをアサインされたユーザーはインスタンスの招待コードを発行できるように @syuilo | ||||
| - 非モデレーターでも、権限を持つロールをアサインされたユーザーはカスタム絵文字の追加、編集、削除を行えるように @syuilo | ||||
| - クリップおよびクリップ内のノートの作成可能数を設定可能に @syuilo | ||||
| - ユーザーリストおよびユーザーリスト内のユーザーの作成可能数を設定可能に @syuilo | ||||
| - ハードワードミュートの最大文字数を設定可能に @syuilo | ||||
| - Webhookの作成可能数を設定可能に @syuilo | ||||
| - ノートをピン留めできる数を設定可能に @syuilo | ||||
| - Server: signToActivityPubGet is set to true by default @syuilo | ||||
| - Server: improve syslog performance @syuilo | ||||
| - Server: Use undici instead of node-fetch and got @tamaina | ||||
| - Server: Judge instance block by endsWith @tamaina | ||||
| - Server: improve note scoring for featured notes @CyberRex0 | ||||
| - Server: アンケート選択肢の文字数制限を緩和 @syuilo | ||||
| - Server: プロフィールの文字数制限を緩和 @syuilo | ||||
| - Server: add rate limits for some endpoints @syuilo | ||||
| - Server: improve stats api performance @syuilo | ||||
| - Server: improve nodeinfo performance @syuilo | ||||
| - Server: delete outdated notifications regularly to improve db performance @syuilo | ||||
| @@ -73,6 +119,7 @@ You should also include the user name that made the change. | ||||
| - Client: Add link to user RSS feed in profile menu @ssmucny | ||||
| - Client: Compress non-animated PNG files @saschanaz | ||||
| - Client: YouTube window player @sim1222 | ||||
| - Client: show readable error when rate limit exceeded @syuilo | ||||
| - Client: enhance dashboard of control panel @syuilo | ||||
| - Client: Vite is upgraded to v4 @syuilo, @tamaina | ||||
| - Client: HMR is available while yarn dev @tamaina | ||||
| @@ -91,13 +138,16 @@ You should also include the user name that made the change. | ||||
| - Client: add heatmap of daily active users to about page @syuilo | ||||
| - Client: introduce fluent emoji @syuilo | ||||
| - Client: add new theme @syuilo | ||||
| - Client: add new mfm function (position, fg, bg) @syuilo | ||||
| - Client: show fireworks when visit user who today is birthday @syuilo | ||||
| - Client: show bot warning on screen when logged in as bot account @syuilo | ||||
| - Client: AiScriptからカスタム絵文字一覧を参照できるように @syuilo | ||||
| - Client: improve overall performance of client @syuilo | ||||
| - Client: ui tweaks @syuilo | ||||
| - Client: clicker game @syuilo | ||||
|  | ||||
| ### Bugfixes | ||||
| - Server: Fix @tensorflow/tfjs-core's MODULE_NOT_FOUND error @ikuradon | ||||
| - Server: 引用内の文章がnyaizeされてしまう問題を修正 @kabo2468 | ||||
| - Server: Bug fix for Pinned Users lookup on instance @squidicuzz | ||||
| - Server: Fix peers API returning suspended instances @ineffyble | ||||
| @@ -110,6 +160,9 @@ You should also include the user name that made the change. | ||||
| - Server: 特定のPNG画像のアップロードに失敗する問題を修正 @usbharu | ||||
| - Server: 非公開のクリップのURLでOGPレンダリングされる問題を修正 @syuilo | ||||
| - Server: アンテナタイムライン(ストリーミング)が、フォローしていないユーザーの鍵投稿も拾ってしまう @syuilo | ||||
| - Server: follow request list api pagination @sim1222 | ||||
| - Server: ドライブ容量超過時のエラーが適切にレスポンスされない問題を修正 @syuilo | ||||
| - Client: パスワードマネージャーなどでユーザー名がオートコンプリートされない問題を修正 @massongit | ||||
| - Client: 日付形式の文字列などがカスタム絵文字として表示されるのを修正 @syuilo | ||||
| - Client: case insensitive emoji search @saschanaz | ||||
| - Client: 画面の幅が狭いとウィジェットドロワーを閉じる手段がなくなるのを修正 @syuilo | ||||
| @@ -121,6 +174,11 @@ You should also include the user name that made the change. | ||||
| - Client: チャートのツールチップが画面に残ることがあるのを修正 @syuilo | ||||
| - Client: fix wrong link in tutorial @syuilo | ||||
|  | ||||
| ### Special thanks | ||||
| - All contributors | ||||
| - All who have created instances for the beta test | ||||
| - All who participated in the beta test | ||||
|  | ||||
| ## 12.119.1 (2022/12/03) | ||||
| ### Bugfixes | ||||
| - Server: Mitigate AP reference chain DoS vector @skehmatics | ||||
|   | ||||
							
								
								
									
										2
									
								
								COPYING
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								COPYING
									
									
									
									
									
								
							| @@ -1,5 +1,5 @@ | ||||
| Unless otherwise stated this repository is | ||||
| Copyright © 2014-2022 syuilo and contributers | ||||
| Copyright © 2014-2023 syuilo and contributers | ||||
|  | ||||
| And is distributed under The GNU Affero General Public License Version 3, you should have received a copy of the license file as LICENSE. | ||||
|  | ||||
|   | ||||
							
								
								
									
										44
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						
									
										44
									
								
								Dockerfile
									
									
									
									
									
								
							| @@ -1,45 +1,55 @@ | ||||
| FROM node:18.13.0-bullseye AS builder | ||||
| ARG NODE_VERSION=18.13.0-bullseye | ||||
|  | ||||
| ARG NODE_ENV=production | ||||
| FROM node:${NODE_VERSION} AS builder | ||||
|  | ||||
| RUN apt-get update \ | ||||
| 	&& apt-get install -y --no-install-recommends \ | ||||
| 	build-essential | ||||
|  | ||||
| RUN corepack enable | ||||
|  | ||||
| WORKDIR /misskey | ||||
|  | ||||
| COPY [".yarnrc.yml", "package.json", "yarn.lock", "./"] | ||||
| COPY [".yarn", "./.yarn"] | ||||
| COPY ["pnpm-lock.yaml", "pnpm-workspace.yaml", "package.json", "./"] | ||||
| COPY ["scripts", "./scripts"] | ||||
| COPY ["packages/backend/package.json", "./packages/backend/"] | ||||
| COPY ["packages/frontend/package.json", "./packages/frontend/"] | ||||
| COPY ["packages/sw/package.json", "./packages/sw/"] | ||||
|  | ||||
| RUN yarn install --immutable | ||||
| RUN pnpm i --frozen-lockfile | ||||
|  | ||||
| COPY . ./ | ||||
|  | ||||
| ARG NODE_ENV=production | ||||
|  | ||||
| RUN git submodule update --init | ||||
| RUN yarn build | ||||
| RUN pnpm build | ||||
|  | ||||
| FROM node:18.13.0-bullseye-slim AS runner | ||||
| FROM node:${NODE_VERSION}-slim AS runner | ||||
|  | ||||
| WORKDIR /misskey | ||||
| ARG UID="991" | ||||
| ARG GID="991" | ||||
|  | ||||
| RUN apt-get update \ | ||||
| 	&& apt-get install -y --no-install-recommends \ | ||||
| 	ffmpeg tini \ | ||||
| 	&& apt-get -y clean \ | ||||
| 	&& rm -rf /var/lib/apt/lists/* | ||||
| 	&& rm -rf /var/lib/apt/lists/* \ | ||||
| 	&& corepack enable \ | ||||
| 	&& groupadd -g "${GID}" misskey \ | ||||
| 	&& useradd -l -u "${UID}" -g "${GID}" -m -d /misskey misskey | ||||
|  | ||||
| COPY --from=builder /misskey/.yarn/install-state.gz ./.yarn/install-state.gz | ||||
| COPY --from=builder /misskey/node_modules ./node_modules | ||||
| COPY --from=builder /misskey/built ./built | ||||
| COPY --from=builder /misskey/packages/backend/node_modules ./packages/backend/node_modules | ||||
| COPY --from=builder /misskey/packages/backend/built ./packages/backend/built | ||||
| COPY --from=builder /misskey/packages/frontend/node_modules ./packages/frontend/node_modules | ||||
| COPY . ./ | ||||
| USER misskey | ||||
| WORKDIR /misskey | ||||
|  | ||||
| COPY --chown=misskey:misskey --from=builder /misskey/node_modules ./node_modules | ||||
| COPY --chown=misskey:misskey --from=builder /misskey/built ./built | ||||
| COPY --chown=misskey:misskey --from=builder /misskey/packages/backend/node_modules ./packages/backend/node_modules | ||||
| COPY --chown=misskey:misskey --from=builder /misskey/packages/backend/built ./packages/backend/built | ||||
| COPY --chown=misskey:misskey --from=builder /misskey/packages/frontend/node_modules ./packages/frontend/node_modules | ||||
| COPY --chown=misskey:misskey --from=builder /misskey/fluent-emojis /misskey/fluent-emojis | ||||
| COPY --chown=misskey:misskey . ./ | ||||
|  | ||||
| ENV NODE_ENV=production | ||||
| ENTRYPOINT ["/usr/bin/tini", "--"] | ||||
| CMD ["yarn", "run", "migrateandstart"] | ||||
| CMD ["pnpm", "run", "migrateandstart"] | ||||
|   | ||||
| @@ -29,17 +29,17 @@ describe('After user signed in', () => { | ||||
|  | ||||
| 	it('first widget should be removed', () => { | ||||
| 		cy.get('.mk-widget-edit').click(); | ||||
| 		cy.get('.customize-container:first-child .remove._button').click(); | ||||
| 		cy.get('.customize-container').should('have.length', 2); | ||||
| 		cy.get('.data-cy-customize-container:first-child .data-cy-customize-container-remove._button').click(); | ||||
| 		cy.get('.data-cy-customize-container').should('have.length', 2); | ||||
| 	}); | ||||
|  | ||||
| 	function buildWidgetTest(widgetName) { | ||||
| 		it(`${widgetName} widget should get added`, () => { | ||||
| 			cy.get('.mk-widget-edit').click(); | ||||
| 			cy.get('.mk-widget-select select').select(widgetName, { force: true }); | ||||
| 			cy.get('.bg._modalBg.transparent').click({ multiple: true, force: true }); | ||||
| 			cy.get('.data-cy-bg._modalBg.data-cy-transparent').click({ multiple: true, force: true }); | ||||
| 			cy.get('.mk-widget-add').click({ force: true }); | ||||
| 			cy.get(`.mkw-${widgetName}`).should('exist'); | ||||
| 			cy.get(`.data-cy-mkw-${widgetName}`).should('exist'); | ||||
| 		}); | ||||
| 	} | ||||
|  | ||||
|   | ||||
| @@ -8,6 +8,11 @@ services: | ||||
|       - db | ||||
|       - redis | ||||
| #      - es | ||||
|     depends_on: | ||||
|       db: | ||||
|         condition: service_healthy | ||||
|       redis: | ||||
|         condition: service_healthy | ||||
|     ports: | ||||
|       - "3000:3000" | ||||
|     networks: | ||||
| @@ -19,21 +24,29 @@ services: | ||||
|  | ||||
|   redis: | ||||
|     restart: always | ||||
|     image: redis:4.0-alpine | ||||
|     image: redis:7-alpine | ||||
|     networks: | ||||
|       - internal_network | ||||
|     volumes: | ||||
|       - ./redis:/data | ||||
|     healthcheck: | ||||
|       test: "redis-cli ping" | ||||
|       interval: 5s | ||||
|       retries: 20 | ||||
|  | ||||
|   db: | ||||
|     restart: always | ||||
|     image: postgres:12.2-alpine | ||||
|     image: postgres:15-alpine | ||||
|     networks: | ||||
|       - internal_network | ||||
|     env_file: | ||||
|       - .config/docker.env | ||||
|     volumes: | ||||
|       - ./db:/var/lib/postgresql/data | ||||
|     healthcheck: | ||||
|       test: "pg_isready -U $$POSTGRES_USER -d $$POSTGRES_DB" | ||||
|       interval: 5s | ||||
|       retries: 20 | ||||
|  | ||||
| #  es: | ||||
| #    restart: always | ||||
|   | ||||
| @@ -817,6 +817,13 @@ account: "الحسابات" | ||||
| cannotLoad: "تعذر التحميل" | ||||
| like: "أعجبني" | ||||
| show: "المظهر" | ||||
| color: "اللون" | ||||
| _role: | ||||
|   priority: "الأولوية" | ||||
|   _priority: | ||||
|     low: "منخفضة" | ||||
|     middle: "متوسط" | ||||
|     high: "عالية" | ||||
| _emailUnavailable: | ||||
|   used: "هذا البريد الإلكتروني مستخدم" | ||||
|   format: "صيغة البريد الإلكتروني غير صالحة" | ||||
| @@ -840,6 +847,7 @@ _accountDelete: | ||||
| _ad: | ||||
|   back: "رجوع" | ||||
|   reduceFrequencyOfThisAd: "قلل عرض هذا الإعلان" | ||||
|   hide: "لا تظهره بتاتًا" | ||||
| _forgotPassword: | ||||
|   enterEmail: "أدخل البريد الإلكتروني المرتبط بحسابك لكي يرسل إليك رابط لإعادة تعيين كلمة المرور." | ||||
|   ifNoEmail: "إذا لم تربط حسابك ببريد إلكتروني سيتوجب عليك التواصل مع مدير الموقع." | ||||
|   | ||||
| @@ -853,6 +853,13 @@ localOnly: "শুধুমাত্র লোকাল" | ||||
| account: "অ্যাকাউন্টগুলি" | ||||
| like: "পছন্দ করা" | ||||
| show: "প্রদর্শন" | ||||
| color: "রং" | ||||
| _role: | ||||
|   priority: "অগ্রাধিকার" | ||||
|   _priority: | ||||
|     low: "নিম্ন" | ||||
|     middle: "মাঝারি" | ||||
|     high: "উচ্চ" | ||||
| _emailUnavailable: | ||||
|   used: "এই ইমেইল ঠিকানাটি ইতোমধ্যে ব্যবহৃত হয়েছে" | ||||
|   format: "এই ইমেল ঠিকানাটি সঠিকভাবে লিখা হয়নি" | ||||
| @@ -877,6 +884,7 @@ _accountDelete: | ||||
| _ad: | ||||
|   back: "পিছনে" | ||||
|   reduceFrequencyOfThisAd: "এই বিজ্ঞাপনটি কম দেখান" | ||||
|   hide: "দেখাবেন না" | ||||
| _forgotPassword: | ||||
|   enterEmail: "আপনি আপনার অ্যাকাউন্টের জন্য নিবন্ধিত ইমেল ঠিকানা লিখুন. সেই ঠিকানায় একটি পাসওয়ার্ড রিসেট লিঙ্ক পাঠানো হবে।" | ||||
|   ifNoEmail: "আপনি যদি নিবন্ধনের সময় ই-মেইল ঠিকানা না দিয়ে থাকেন, তাহলে অনুগ্রহ করে প্রশাসকের সাথে যোগাযোগ করুন।" | ||||
|   | ||||
| @@ -611,6 +611,13 @@ slow: "Pomalá" | ||||
| fast: "Rychlá" | ||||
| account: "Účty" | ||||
| show: "Zobrazit" | ||||
| color: "Barva" | ||||
| _role: | ||||
|   priority: "Priorita" | ||||
|   _priority: | ||||
|     low: "Nízká" | ||||
|     middle: "Střední" | ||||
|     high: "Vysoká" | ||||
| _ad: | ||||
|   back: "Zpět" | ||||
| _gallery: | ||||
|   | ||||
| @@ -924,6 +924,76 @@ neverShow: "Nicht wieder anzeigen" | ||||
| remindMeLater: "Vielleicht später" | ||||
| didYouLikeMisskey: "Gefällt dir Misskey?" | ||||
| pleaseDonate: "Misskey ist die kostenlose Software, die von {host} verwendet wird. Wir würden uns über Spenden freuen, damit dessen Entwicklung weitergeführt werden kann!" | ||||
| roles: "Rollen" | ||||
| role: "Rolle" | ||||
| normalUser: "Standardbenutzer" | ||||
| undefined: "Undefiniert" | ||||
| assign: "Zuweisen" | ||||
| unassign: "Entfernen" | ||||
| color: "Farbe" | ||||
| manageCustomEmojis: "Kann benutzerdefinierte Emojis verwalten" | ||||
| youCannotCreateAnymore: "Du hast das Erstellungslimit erreicht." | ||||
| cannotPerformTemporary: "Vorübergehend nicht verfügbar" | ||||
| cannotPerformTemporaryDescription: "Diese Aktion ist wegen des Überschreitenes des Ausführungslimits temporär nicht verfügbar. Bitte versuche es nach einiger Zeit erneut." | ||||
| preset: "Vorlage" | ||||
| selectFromPresets: "Aus Vorlagen wählen" | ||||
| _role: | ||||
|   new: "Rolle erstellen" | ||||
|   edit: "Rolle bearbeiten" | ||||
|   name: "Rollenname" | ||||
|   description: "Rollenbeschreibung" | ||||
|   permission: "Rollenberechtigungen" | ||||
|   descriptionOfPermission: "<b>Moderatoren</b> können grundlegende Verwaltungsaufgaben erledigen.\n<b>Administratoren</b> können alle Einstellungen der Instanz verwalten." | ||||
|   assignTarget: "Zuweisungsart" | ||||
|   descriptionOfAssignTarget: "<b>Manuell</b> bedeutet, dass die Liste der Benutzer einer Rolle manuell verwaltet wird.\n<b>Konditional</b> bedeutet, dass die Liste der Benutzer einer Rolle durch eine Bedingung automatisch verwaltet wird." | ||||
|   manual: "Manuell" | ||||
|   conditional: "Konditional" | ||||
|   condition: "Bedingung" | ||||
|   isConditionalRole: "Dies ist eine konditionale Rolle." | ||||
|   isPublic: "Öffentliche Rolle" | ||||
|   descriptionOfIsPublic: "Ist dies aktiviert, so kann jeder die Liste der Benutzer, die dieser Rolle zugewiesen sind, einsehen. Zusätzlich wird diese Rolle im Profil zugewiesener Benutzer angezeigt." | ||||
|   options: "Optionen" | ||||
|   policies: "Richtlinien" | ||||
|   baseRole: "Rollenvorlage" | ||||
|   useBaseValue: "Wert der Rollenvorlage verwenden" | ||||
|   chooseRoleToAssign: "Zuzuweisende Rolle auswählen" | ||||
|   canEditMembersByModerator: "Moderatoren können Benutzern diese Rolle zuweisen" | ||||
|   descriptionOfCanEditMembersByModerator: "Wenn aktiviert, so können Moderatoren und Adminstratoren anderen Benutzern diese Rolle zuweisen bzw. diese Zuweisung aufheben. Wenn deaktiviert, so ist es nur Administratoren möglich, Zuweisungen dieser Rolle zu verwalten." | ||||
|   priority: "Priorität" | ||||
|   _priority: | ||||
|     low: "Niedrig" | ||||
|     middle: "Mittel" | ||||
|     high: "Hoch" | ||||
|   _options: | ||||
|     gtlAvailable: "Kann auf die globale Chronik zugreifen" | ||||
|     ltlAvailable: "Kann auf die lokale Chronik zugreifen" | ||||
|     canPublicNote: "Kann öffentliche Notizen erstellen" | ||||
|     canInvite: "Kann Einladungscodes für diese Instanz erstellen" | ||||
|     canManageCustomEmojis: "Benutzerdefinierte Emojis verwalten" | ||||
|     driveCapacity: "Drive-Kapazität" | ||||
|     pinMax: "Maximale Anzahl an angehefteten Notizen" | ||||
|     antennaMax: "Maximale Anzahl an Antennen" | ||||
|     wordMuteMax: "Maximale Zeichenlänge für Wortstummschaltungen" | ||||
|     webhookMax: "Maximale Anzahl an Webhooks" | ||||
|     clipMax: "Maximale Anzahl an Clips" | ||||
|     noteEachClipsMax: "Maximale Anzahl an Notizen innerhalb eines Clips" | ||||
|     userListMax: "Maximale Anzahl an Benutzern in einer Benutzerliste" | ||||
|     userEachUserListsMax: "Maximale Anzahl an Benutzerlisten" | ||||
|     rateLimitFactor: "Versuchsanzahl" | ||||
|     descriptionOfRateLimitFactor: "Je niedriger desto weniger restriktiv, je höher destro restriktiver." | ||||
|     canHideAds: "Kann Werbung ausblenden" | ||||
|   _condition: | ||||
|     isLocal: "Lokaler Benutzer" | ||||
|     isRemote: "Benutzer fremder Instanz" | ||||
|     createdLessThan: "Kontoerstellung liegt weniger als X zurück" | ||||
|     createdMoreThan: "Kontoerstellung liegt mehr als X zurück" | ||||
|     followersLessThanOrEq: "Hat X oder weniger Follower" | ||||
|     followersMoreThanOrEq: "Hat X oder mehr Follower" | ||||
|     followingLessThanOrEq: "Folgt X oder weniger Benutzern" | ||||
|     followingMoreThanOrEq: "Folgt X oder mehr Benutzern" | ||||
|     and: "UND-Bedingung" | ||||
|     or: "ODER-Bedingung" | ||||
|     not: "NICHT-Bedingung" | ||||
| _sensitiveMediaDetection: | ||||
|   description: "Ermöglicht eine Erleichterung der Servermoderation durch die automatische Erkennungen von NSFW-Medien unter Verwendung von Machine Learning. Hierdurch wird die Serverlast etwas erhöht." | ||||
|   sensitivity: "Erkennungssensitivität" | ||||
| @@ -956,6 +1026,7 @@ _accountDelete: | ||||
| _ad: | ||||
|   back: "Zurück" | ||||
|   reduceFrequencyOfThisAd: "Diese Werbung weniger anzeigen" | ||||
|   hide: "Ausblenden" | ||||
| _forgotPassword: | ||||
|   enterEmail: "Gib die Email-Adresse ein, mit der du dich registriert hast. An diese wird ein Link gesendet, mit dem du dein Passwort zurücksetzen kannst." | ||||
|   ifNoEmail: "Solltest du bei der Registrierung keine Email-Adresse angegeben haben, wende dich bitte an den Administrator." | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| --- | ||||
| _lang_: "Ελληνικά" | ||||
| monthAndDay: "{μήνας}/{ημέρα}" | ||||
| monthAndDay: "{day}/{month}" | ||||
| search: "Αναζήτηση" | ||||
| notifications: "Ειδοποιήσεις" | ||||
| username: "Όνομα μέλους" | ||||
|   | ||||
| @@ -924,6 +924,76 @@ neverShow: "Don't show again" | ||||
| remindMeLater: "Maybe later" | ||||
| didYouLikeMisskey: "Have you taken a liking to Misskey?" | ||||
| pleaseDonate: "{host} uses the free software, Misskey. We would highly appreciate your donations so development of Misskey can continue!" | ||||
| roles: "Roles" | ||||
| role: "Role" | ||||
| normalUser: "Normal user" | ||||
| undefined: "Undefined" | ||||
| assign: "Assign" | ||||
| unassign: "Unassign" | ||||
| color: "Color" | ||||
| manageCustomEmojis: "Manage Custom Emojis" | ||||
| youCannotCreateAnymore: "You've hit the creation limit." | ||||
| cannotPerformTemporary: "Temporarily unavailable" | ||||
| cannotPerformTemporaryDescription: "This action cannot be performed temporarily due to exceeding the execution limit. Please wait for a while and then try again." | ||||
| preset: "Preset" | ||||
| selectFromPresets: "Choose from presets" | ||||
| _role: | ||||
|   new: "New role" | ||||
|   edit: "Edit role" | ||||
|   name: "Role name" | ||||
|   description: "Role description" | ||||
|   permission: "Role permissions" | ||||
|   descriptionOfPermission: "<b>Moderators</b> can perform basic moderation operations.\n<b>Administrators</b> can change all settings of the instance." | ||||
|   assignTarget: "Assignment type" | ||||
|   descriptionOfAssignTarget: "<b>Manual</b> to manually change who is part of this role and who is not.\n<b>Conditional</b> to have users be automatically assigned and removed from this role based on a condition." | ||||
|   manual: "Manual" | ||||
|   conditional: "Conditional" | ||||
|   condition: "Condition" | ||||
|   isConditionalRole: "This is a conditional role." | ||||
|   isPublic: "Public role" | ||||
|   descriptionOfIsPublic: "Anyone will be able to view a list of users assigned to this role. In addition, this role will be displayed in the profiles of assigned users." | ||||
|   options: "Role options" | ||||
|   policies: "Policies" | ||||
|   baseRole: "Role template" | ||||
|   useBaseValue: "Use role template value" | ||||
|   chooseRoleToAssign: "Select the role to assign" | ||||
|   canEditMembersByModerator: "Allow moderators to edit the list of members for this role" | ||||
|   descriptionOfCanEditMembersByModerator: "When turned on, moderators as well as administrators will be able to assign and unassign users to this role. When turned off, only administrators will be able to assign users." | ||||
|   priority: "Priority" | ||||
|   _priority: | ||||
|     low: "Low" | ||||
|     middle: "Medium" | ||||
|     high: "High" | ||||
|   _options: | ||||
|     gtlAvailable: "Can view the global timeline" | ||||
|     ltlAvailable: "Can view the local timeline" | ||||
|     canPublicNote: "Can send public notes" | ||||
|     canInvite: "Can create instance invite codes" | ||||
|     canManageCustomEmojis: "Can manage custom emojis" | ||||
|     driveCapacity: "Drive capacity" | ||||
|     pinMax: "Maximum number of pinned notes" | ||||
|     antennaMax: "Maximum number of antennas" | ||||
|     wordMuteMax: "Maximum number of characters allowed in word mutes" | ||||
|     webhookMax: "Maximum number of Webhooks" | ||||
|     clipMax: "Maximum number of Clips" | ||||
|     noteEachClipsMax: "Maximum number of notes within a clip" | ||||
|     userListMax: "Maximum number of user lists" | ||||
|     userEachUserListsMax: "Maximum number of users within a user list" | ||||
|     rateLimitFactor: "Rate limit" | ||||
|     descriptionOfRateLimitFactor: "Lower rate limits are less restrictive, higher ones more restrictive. " | ||||
|     canHideAds: "Can hide ads" | ||||
|   _condition: | ||||
|     isLocal: "Local user" | ||||
|     isRemote: "Remote user" | ||||
|     createdLessThan: "Less than X has passed since account creation" | ||||
|     createdMoreThan: "More than X has passed since account creation" | ||||
|     followersLessThanOrEq: "Has X or fewer followers" | ||||
|     followersMoreThanOrEq: "Has X or more followers" | ||||
|     followingLessThanOrEq: "Follows X or fewer accounts" | ||||
|     followingMoreThanOrEq: "Follows X or more accounts" | ||||
|     and: "AND-Condition" | ||||
|     or: "OR-Condition" | ||||
|     not: "NOT-Condition" | ||||
| _sensitiveMediaDetection: | ||||
|   description: "Reduces the effort of server moderation through automatically recognizing NSFW media via Machine Learning. This will slightly increase the load on the server." | ||||
|   sensitivity: "Detection sensitivity" | ||||
| @@ -956,6 +1026,7 @@ _accountDelete: | ||||
| _ad: | ||||
|   back: "Back" | ||||
|   reduceFrequencyOfThisAd: "Show this ad less" | ||||
|   hide: "Hide" | ||||
| _forgotPassword: | ||||
|   enterEmail: "Enter the email address you used to register. A link with which you can reset your password will then be sent to it." | ||||
|   ifNoEmail: "If you did not use an email during registration, please contact the instance administrator instead." | ||||
|   | ||||
| @@ -918,6 +918,13 @@ cannotLoad: "No se puede cargar." | ||||
| numberOfProfileView: "Número de vistas de perfil" | ||||
| like: "¡Muy bien!" | ||||
| show: "Apariencia" | ||||
| color: "Color" | ||||
| _role: | ||||
|   priority: "Prioridad" | ||||
|   _priority: | ||||
|     low: "Baja" | ||||
|     middle: "Mediano" | ||||
|     high: "Alta" | ||||
| _sensitiveMediaDetection: | ||||
|   description: "Reduce el esfuerzo de la moderación el el servidor a través del reconocimiento automático de contenido NSFW usando 'Machine Learning'. Esto puede incrementar ligeramente la carga en el servidor." | ||||
|   sensitivity: "Sensibilidad de detección" | ||||
| @@ -950,6 +957,7 @@ _accountDelete: | ||||
| _ad: | ||||
|   back: "Deseleccionar" | ||||
|   reduceFrequencyOfThisAd: "Mostrar menos este anuncio." | ||||
|   hide: "No mostrar" | ||||
| _forgotPassword: | ||||
|   enterEmail: "Ingrese el correo usado para registrar la cuenta. Se enviará un link para resetear la contraseña." | ||||
|   ifNoEmail: "Si no utilizó un correo para crear la cuenta, contáctese con el administrador." | ||||
|   | ||||
| @@ -911,7 +911,17 @@ loggedInAsBot: "Connecté actuellement en tant que bot" | ||||
| tools: "Outils" | ||||
| cannotLoad: "Chargement impossible" | ||||
| like: "J'aime" | ||||
| numberOfLikes: "Favoris" | ||||
| show: "Affichage" | ||||
| neverShow: "Ne plus afficher" | ||||
| remindMeLater: "Peut-être plus tard" | ||||
| color: "Couleur" | ||||
| _role: | ||||
|   priority: "Priorité" | ||||
|   _priority: | ||||
|     low: "Basse" | ||||
|     middle: "Moyen" | ||||
|     high: "Haute" | ||||
| _sensitiveMediaDetection: | ||||
|   description: "L'apprentissage automatique peut être utilisé pour détecter automatiquement les médias sensibles à modérer. La sollicitation des serveurs augmente légèrement." | ||||
|   sensitivity: "Sensibilité de la détection" | ||||
| @@ -944,6 +954,7 @@ _accountDelete: | ||||
| _ad: | ||||
|   back: "Retour" | ||||
|   reduceFrequencyOfThisAd: "Voir cette publicité moins souvent" | ||||
|   hide: "Cacher " | ||||
| _forgotPassword: | ||||
|   enterEmail: "Entrez ici l'adresse e-mail que vous avez enregistrée pour votre compte. Un lien vous permettant de réinitialiser votre mot de passe sera envoyé à cette adresse." | ||||
|   ifNoEmail: "Si vous n'avez pas enregistré d'adresse e-mail, merci de contacter l'administrateur·rice de votre instance." | ||||
|   | ||||
| @@ -2,6 +2,7 @@ | ||||
| _lang_: "Bahasa Indonesia" | ||||
| headlineMisskey: "Jaringan terhubung melalui catatan" | ||||
| introMisskey: "Selamat datang! Misskey adalah perangkat mikroblog tercatu bersifat sumber terbuka.\nMulailah menuliskan catatan, bagikan peristiwa terkini, serta ceritakan segala tentangmu.📡\nTunjukkan juga reaksimu pada catatan pengguna lain.👍\nMari jelajahi dunia baru🚀" | ||||
| poweredByMisskeyDescription: "{name} adalah sebuah layanan (instance) yang menggunakan platform sumber terbuka <b>Misskey</b>." | ||||
| monthAndDay: "{day} {month}" | ||||
| search: "Penelusuran" | ||||
| notifications: "Pemberitahuan" | ||||
| @@ -47,6 +48,7 @@ deleteAndEdit: "Hapus dan sunting" | ||||
| deleteAndEditConfirm: "Apakah kamu yakin ingin menghapus note ini dan menyuntingnya? Kamu akan kehilangan semua reaksi, renote dan balasan di note ini." | ||||
| addToList: "Tambahkan ke daftar" | ||||
| sendMessage: "Kirim pesan" | ||||
| copyRSS: "Salin RSS" | ||||
| copyUsername: "Salin nama pengguna" | ||||
| searchUser: "Cari pengguna" | ||||
| reply: "Balas" | ||||
| @@ -383,6 +385,7 @@ administrator: "Admin" | ||||
| token: "Token" | ||||
| twoStepAuthentication: "Otentikasi dua faktor" | ||||
| moderator: "Moderator" | ||||
| moderation: "Moderasi" | ||||
| nUsersMentioned: "{n} pengguna disebut" | ||||
| securityKey: "Kunci keamanan" | ||||
| securityKeyName: "Nama kunci" | ||||
| @@ -449,6 +452,7 @@ language: "Bahasa" | ||||
| uiLanguage: "Bahasa antarmuka pengguna" | ||||
| groupInvited: "Telah diundang ke grup" | ||||
| aboutX: "Tentang {x}" | ||||
| emojiStyle: "Gaya emoji" | ||||
| disableDrawer: "Jangan gunakan menu bergaya laci" | ||||
| youHaveNoGroups: "Kamu tidak memiliki grup" | ||||
| joinOrCreateGroup: "Bergabunglah dengan grup atau kamu dapat membuat grupmu sendiri." | ||||
| @@ -561,6 +565,7 @@ author: "Pembuat" | ||||
| leaveConfirm: "Ada perubahan yang belum disimpan. Apakah kamu ingin membuangnya?" | ||||
| manage: "Manajemen" | ||||
| plugins: "Plugin" | ||||
| preferencesBackups: "Aturan pencadangan" | ||||
| deck: "Dek" | ||||
| undeck: "Keluar dari dek" | ||||
| useBlurEffectForModal: "Gunakan efek buram untuk modal" | ||||
| @@ -706,6 +711,7 @@ accentColor: "Aksen" | ||||
| textColor: "Teks" | ||||
| saveAs: "Simpan sebagai…" | ||||
| advanced: "Tingkat lanjut" | ||||
| advancedSettings: "Pengaturan Lanjut" | ||||
| value: "Nilai" | ||||
| createdAt: "Dibuat pada" | ||||
| updatedAt: "Diperbarui pada" | ||||
| @@ -850,15 +856,33 @@ rateLimitExceeded: "Batas sudah terlampaui" | ||||
| cropImage: "potong gambar" | ||||
| cropImageAsk: "Ingin memotong gambar?" | ||||
| file: "Berkas" | ||||
| noEmailServerWarning: "Mail Server tidak disetel." | ||||
| recommended: "Disarankan" | ||||
| check: "Cek" | ||||
| deleteAccount: "Hapus Akun" | ||||
| logoutConfirm: "Anda yakin ingin keluar?" | ||||
| lastActiveDate: "Terakhir digunakan" | ||||
| statusbar: "Bilah status" | ||||
| pleaseSelect: "Pilih opsi..." | ||||
| reverse: "Balik" | ||||
| colored: "Diwarnai" | ||||
| refreshInterval: "Jeda pembaharuan" | ||||
| label: "Label" | ||||
| type: "Tipe" | ||||
| localOnly: "Hanya lokal" | ||||
| shuffle: "Acak" | ||||
| account: "Akun" | ||||
| like: "Suka" | ||||
| unlike: "Tidak Suka" | ||||
| numberOfLikes: "Jumlah yang disukai" | ||||
| show: "Tampilkan" | ||||
| color: "Warna" | ||||
| _role: | ||||
|   priority: "Prioritas" | ||||
|   _priority: | ||||
|     low: "Rendah" | ||||
|     middle: "Sedang" | ||||
|     high: "Tinggi" | ||||
| _emailUnavailable: | ||||
|   used: "Alamat surel ini telah digunakan" | ||||
|   format: "Format tidak valid." | ||||
| @@ -883,6 +907,7 @@ _accountDelete: | ||||
| _ad: | ||||
|   back: "Kembali" | ||||
|   reduceFrequencyOfThisAd: "Tampilkan iklan ini lebih sedikit" | ||||
|   hide: "Jangan tampilkan" | ||||
| _forgotPassword: | ||||
|   enterEmail: "Masukkan alamat surel yang kamu gunakan pada saat mendaftar. Sebuah tautan untuk mengatur ulang kata sandi kamu akan dikirimkan ke alamat surel tersebut." | ||||
|   ifNoEmail: "Apabila kamu tidak menggunakan surel pada saat pendaftaran, mohon hubungi admin segera." | ||||
|   | ||||
| @@ -8,7 +8,7 @@ search: "Cerca" | ||||
| notifications: "Notifiche" | ||||
| username: "Nome utente" | ||||
| password: "Password" | ||||
| forgotPassword: "Hai dimenticato la tua password?" | ||||
| forgotPassword: "Hai dimenticato la password?" | ||||
| fetchingAsApObject: "Recuperando dal Fediverso..." | ||||
| ok: "OK" | ||||
| gotIt: "Ho capito" | ||||
| @@ -109,7 +109,7 @@ you: "Tu" | ||||
| clickToShow: "Clicca per visualizzare" | ||||
| sensitive: "Contenuto sensibile" | ||||
| add: "Aggiungi" | ||||
| reaction: "Reazione" | ||||
| reaction: "Reazioni" | ||||
| reactionSetting: "Reazioni visualizzate sul pannello" | ||||
| reactionSettingDescription2: "Trascina per riorganizzare, clicca per cancellare, usa il pulsante \"+\" per aggiungere." | ||||
| rememberNoteVisibility: "Ricordare le impostazioni di visibilità delle note" | ||||
| @@ -226,7 +226,7 @@ currentPassword: "Password attuale" | ||||
| newPassword: "Nuova Password" | ||||
| newPasswordRetype: "Conferma password" | ||||
| attachFile: "Allega file" | ||||
| more: "Altri!" | ||||
| more: "Di più!" | ||||
| featured: "Tendenze" | ||||
| usernameOrUserId: "Nome utente o ID utente" | ||||
| noSuchUser: "Nessun utente trovato" | ||||
| @@ -325,7 +325,7 @@ connectService: "Connessione" | ||||
| disconnectService: "Disconnessione " | ||||
| enableLocalTimeline: "Abilita Timeline locale" | ||||
| enableGlobalTimeline: "Abilita Timeline federata" | ||||
| disablingTimelinesInfo: "Anche se disabiliti queste timeline, gli amministratori e i moderatori potranno sempre accederci." | ||||
| disablingTimelinesInfo: "Anche disabilitandole, gli Amministratori e i Moderatori potranno comunque accedervi." | ||||
| registration: "Iscriviti" | ||||
| enableRegistration: "Permettere nuove registrazioni" | ||||
| invite: "Invita" | ||||
| @@ -340,7 +340,7 @@ pinnedUsers: "Utenti in evidenza" | ||||
| pinnedUsersDescription: "Elenca gli/le utenti che vuoi fissare in cima alla pagina \"Esplora\", un@ per riga." | ||||
| pinnedPages: "Pagine in evidenza" | ||||
| pinnedPagesDescription: "Specifica il percorso delle pagine che vuoi fissare in cima alla pagina dell'istanza. Una pagina per riga." | ||||
| pinnedClipId: "ID della nota in evidenza" | ||||
| pinnedClipId: "ID della Clip in evidenza" | ||||
| pinnedNotes: "Nota fissata" | ||||
| hcaptcha: "hCaptcha" | ||||
| enableHcaptcha: "Abilita hCaptcha" | ||||
| @@ -512,7 +512,7 @@ newNoteRecived: "Vedi le nuove note" | ||||
| sounds: "Impostazioni suoni" | ||||
| sound: "Impostazioni suoni" | ||||
| listen: "Ascolta" | ||||
| none: "Niente" | ||||
| none: "Nessuno" | ||||
| showInPage: "Visualizza in pagina" | ||||
| popout: "Finestra pop-out" | ||||
| volume: "Volume" | ||||
| @@ -578,7 +578,7 @@ useFullReactionPicker: "Usa la totalità del pannello di reazioni" | ||||
| width: "Larghezza" | ||||
| height: "Altezza" | ||||
| large: "Grande" | ||||
| medium: "Predefinito" | ||||
| medium: "Medio" | ||||
| small: "Piccolo" | ||||
| generateAccessToken: "Genera token di accesso" | ||||
| permission: "Autorizzazioni " | ||||
| @@ -649,13 +649,13 @@ instanceTicker: "Informazioni sull'istanza da cui vengono le note" | ||||
| waitingFor: "Aspettando {x}" | ||||
| random: "Casuale" | ||||
| system: "Sistema" | ||||
| switchUi: "Cambiare interfaccia utente" | ||||
| switchUi: "Cambiare interfaccia" | ||||
| desktop: "Desktop" | ||||
| clip: "Nota" | ||||
| clip: "Clip" | ||||
| createNew: "Crea" | ||||
| optional: "Opzionale" | ||||
| createNewClip: "Nuova Nota" | ||||
| unclip: "Rimuovi la nota" | ||||
| optional: "facoltativo" | ||||
| createNewClip: "Crea una Clip" | ||||
| unclip: "Togli Nota dalla Clip" | ||||
| confirmToUnclipAlreadyClippedNote: "Questa nota è già inclusa in \"{name}\". Si desidera escludere la nota?" | ||||
| public: "Pubblica" | ||||
| i18nInfo: "Misskey è tradotto in diverse lingue da volontari. Anche tu puoi contribuire su {link}." | ||||
| @@ -799,7 +799,7 @@ received: "Ricevuto" | ||||
| searchResult: "Risultati della Ricerca" | ||||
| hashtags: "Hashtag" | ||||
| troubleshooting: "Risoluzione problemi" | ||||
| useBlurEffect: "Utilizza effetto sfocatura per l'interfaccia utente" | ||||
| useBlurEffect: "Utilizza effetto sfocatura nell'interfaccia" | ||||
| learnMore: "Più dettagli" | ||||
| misskeyUpdated: "Misskey è stato aggiornato!" | ||||
| whatIsNew: "Visualizza le informazioni sull'aggiornamento" | ||||
| @@ -836,7 +836,7 @@ hide: "Nascondere" | ||||
| leaveGroup: "Esci dal gruppo" | ||||
| leaveGroupConfirm: "Uscire da「{name}」?" | ||||
| useDrawerReactionPickerForMobile: "Mostra sul drawer da dispositivo mobile" | ||||
| welcomeBackWithName: "Eccoti di nuovo, {name}! Ciao!" | ||||
| welcomeBackWithName: "Ciao, {name}! Eccoti di nuovo!" | ||||
| clickToFinishEmailVerification: "Fai click su [{ok}] per completare la verifica dell'indirizzo email." | ||||
| overridedDeviceKind: "Tipo di dispositivo" | ||||
| smartphone: "Smartphone" | ||||
| @@ -917,7 +917,83 @@ tools: "Strumenti" | ||||
| cannotLoad: "Caricamento impossibile" | ||||
| numberOfProfileView: "Visualizzazioni profilo" | ||||
| like: "Mi piace!" | ||||
| unlike: "Non mi piace" | ||||
| numberOfLikes: "Numero di Like" | ||||
| show: "Visualizza" | ||||
| neverShow: "Non mostrare più" | ||||
| remindMeLater: "Rimanda" | ||||
| didYouLikeMisskey: "Ti piace Misskey?" | ||||
| pleaseDonate: "Misskey è il software libero utilizzato su {host}. Offrendo una donazione è più facile continuare a svilupparlo!" | ||||
| roles: "Ruoli" | ||||
| role: "Ruolo" | ||||
| normalUser: "Profilo standard" | ||||
| undefined: "Indefinito" | ||||
| assign: "Assegna" | ||||
| unassign: "Disassegna" | ||||
| color: "Colore" | ||||
| manageCustomEmojis: "Gestisci le emoji personalizzate" | ||||
| youCannotCreateAnymore: "Non puoi creare, hai raggiunto il limite." | ||||
| cannotPerformTemporary: "Indisponibilità temporanea" | ||||
| cannotPerformTemporaryDescription: "L'attività non può essere svolta, poiché si è raggiunto il limite di esecuzioni possibili. Per favore, riprova più tardi." | ||||
| preset: "Preimpostato" | ||||
| selectFromPresets: "Seleziona preimpostato" | ||||
| _role: | ||||
|   new: "Nuovo ruolo" | ||||
|   edit: "Modifica ruolo" | ||||
|   name: "Nome del ruolo" | ||||
|   description: "Descrizione del ruolo" | ||||
|   permission: "Permessi globali del ruolo" | ||||
|   descriptionOfPermission: "<b>Moderatori</b> possono svolgere le attività di moderazione basilari.\n<b>Amministratori</b> possono modificare la configurazione dell'istanza." | ||||
|   assignTarget: "Modalità di assegnazione del ruolo" | ||||
|   descriptionOfAssignTarget: "<b>Manuale</b>: per assegnare manualmente questo ruolo ai profili.\n<b>Condizionale</b>: per assegnare o rimuovere automaticamente questo ruolo ai profili, a precise condizioni." | ||||
|   manual: "Manuale" | ||||
|   conditional: "Condizionale" | ||||
|   condition: "Condizioni" | ||||
|   isConditionalRole: "Questo è un ruolo condizionato" | ||||
|   isPublic: "Ruolo pubblico" | ||||
|   descriptionOfIsPublic: "La lista di profili assegnati a questo ruolo è visibile a chiunque. Inoltre, il nome del ruolo verrà mostrato pubblicamente nei relativi profili." | ||||
|   options: "Opzioni del ruolo" | ||||
|   policies: "Policy" | ||||
|   baseRole: "Ruolo di base" | ||||
|   useBaseValue: "Eredita dal ruolo base" | ||||
|   chooseRoleToAssign: "Seleziona il ruolo da assegnare" | ||||
|   canEditMembersByModerator: "Anche i Moderatori assegnano profili a questo ruolo" | ||||
|   descriptionOfCanEditMembersByModerator: "Se disattivo, potranno farlo solamente gli Amministratori." | ||||
|   priority: "Priorità" | ||||
|   _priority: | ||||
|     low: "Bassa" | ||||
|     middle: "Medio" | ||||
|     high: "Alta" | ||||
|   _options: | ||||
|     gtlAvailable: "Disponibilità della Timeline Federata" | ||||
|     ltlAvailable: "Disponibilità della Timeline Locale" | ||||
|     canPublicNote: "Può scrivere Note con Visibilità Pubblica" | ||||
|     canInvite: "Genera codici di invito all'istanza" | ||||
|     canManageCustomEmojis: "Gestire le emoji personalizzate" | ||||
|     driveCapacity: "Capienza del Drive" | ||||
|     pinMax: "Quantità massima di Note in primo piano" | ||||
|     antennaMax: "Quantità massima di Antenne" | ||||
|     wordMuteMax: "Lunghezza massima del filtro parole" | ||||
|     webhookMax: "Quantità massima di Webhook" | ||||
|     clipMax: "Quantità massima di Clip" | ||||
|     noteEachClipsMax: "Quantità massima di Note nella Clip" | ||||
|     userListMax: "Quantità massima di liste" | ||||
|     userEachUserListsMax: "Quantità massima di profili per lista" | ||||
|     rateLimitFactor: "Limite del rapporto" | ||||
|     descriptionOfRateLimitFactor: "I rapporti più bassi sono meno restrittivi, quelli più alti lo sono di più." | ||||
|     canHideAds: "Può nascondere i banner" | ||||
|   _condition: | ||||
|     isLocal: "Profilo locale" | ||||
|     isRemote: "Profilo remoto" | ||||
|     createdLessThan: "Creato meno di" | ||||
|     createdMoreThan: "Creato più di" | ||||
|     followersLessThanOrEq: "Ha meno di N follower" | ||||
|     followersMoreThanOrEq: "Ha più di N follower" | ||||
|     followingLessThanOrEq: "Segue N profili o meno" | ||||
|     followingMoreThanOrEq: "Segue N profili o più" | ||||
|     and: "E" | ||||
|     or: "O" | ||||
|     not: "NON" | ||||
| _sensitiveMediaDetection: | ||||
|   description: "L'apprendimento automatico può essere utilizzato per individuare automaticamente i media sensibili da moderare. Il carico del server aumenta leggermente." | ||||
|   sensitivity: "Sensibilità di rilevamento" | ||||
| @@ -950,6 +1026,7 @@ _accountDelete: | ||||
| _ad: | ||||
|   back: "Indietro" | ||||
|   reduceFrequencyOfThisAd: "Visualizza questa pubblicità meno spesso" | ||||
|   hide: "Nascondi" | ||||
| _forgotPassword: | ||||
|   enterEmail: "Inserisci l'indirizzo di posta elettronica che hai registrato nel tuo profilo. Il collegamento necessario per ripristinare la password verrà inviato a questo indirizzo." | ||||
|   ifNoEmail: "Se nessun indirizzo e-mail è stato registrato, si prega di contattare l'amministratore·trice dell'istanza." | ||||
| @@ -1090,9 +1167,9 @@ _channel: | ||||
|   usersCount: "{n} partecipanti" | ||||
|   notesCount: "{n} note" | ||||
| _menuDisplay: | ||||
|   sideFull: "laro" | ||||
|   sideIcon: "Orizzontale (icona)" | ||||
|   top: "superficie" | ||||
|   sideFull: "Laterale" | ||||
|   sideIcon: "Laterale (solo icone)" | ||||
|   top: "In alto" | ||||
|   hide: "Nascondere" | ||||
| _wordMute: | ||||
|   muteWords: "Parole da filtrare" | ||||
| @@ -1194,10 +1271,10 @@ _ago: | ||||
|   future: "Futuro" | ||||
|   justNow: "Ora" | ||||
|   secondsAgo: "{n}s fa" | ||||
|   minutesAgo: "{n}min fa" | ||||
|   minutesAgo: "{n} min fa" | ||||
|   hoursAgo: "{n} ore fa" | ||||
|   daysAgo: "{n} giorni fa" | ||||
|   weeksAgo: "{n} settimane fa" | ||||
|   daysAgo: "{n} gg fa" | ||||
|   weeksAgo: "{n} sett. fa" | ||||
|   monthsAgo: "{n} mesi fa" | ||||
|   yearsAgo: "{n} anni fa" | ||||
| _time: | ||||
| @@ -1319,10 +1396,12 @@ _widgets: | ||||
|   jobQueue: "Coda di lavoro" | ||||
|   serverMetric: "Statistiche server" | ||||
|   aiscript: "Console AiScript" | ||||
|   aiscriptApp: "App AiScript" | ||||
|   aichan: "Mascotte Ai" | ||||
|   userList: "Elenco utenti" | ||||
|   _userList: | ||||
|     chooseList: "Seleziona una lista" | ||||
|   clicker: "Cliccaggio" | ||||
| _cw: | ||||
|   hide: "Nascondere" | ||||
|   show: "Mostra di più" | ||||
| @@ -1331,7 +1410,7 @@ _cw: | ||||
| _poll: | ||||
|   noOnlyOneChoice: "Sono necessarie almeno 2 risposte" | ||||
|   choiceN: "Opzione {n}" | ||||
|   noMore: "Hai aggiunto il numero massimo di opzioni." | ||||
|   noMore: "Hai raggiunto il limite di opzioni." | ||||
|   canMultipleVote: "Possibilità di risposte multiple" | ||||
|   expiration: "Scadenza" | ||||
|   infinite: "Non scade" | ||||
| @@ -1425,7 +1504,16 @@ _timelines: | ||||
|   social: "Sociale" | ||||
|   global: "Federata" | ||||
| _play: | ||||
|   new: "Crea un Play" | ||||
|   edit: "Modifica i Play" | ||||
|   created: "Il Play è stato creato" | ||||
|   updated: "Il Play è stato aggiornato" | ||||
|   deleted: "Il Play è stato eliminato" | ||||
|   pageSetting: "Impostazioni di Play" | ||||
|   editThisPage: "Modifica il Play" | ||||
|   viewSource: "Visualizza sorgente" | ||||
|   my: "I miei Play" | ||||
|   liked: "Play piaciuti" | ||||
|   featured: "Popolari" | ||||
|   title: "Titolo" | ||||
|   script: "Script" | ||||
|   | ||||
| @@ -110,6 +110,7 @@ clickToShow: "クリックして表示" | ||||
| sensitive: "閲覧注意" | ||||
| add: "追加" | ||||
| reaction: "リアクション" | ||||
| reactions: "リアクション" | ||||
| reactionSetting: "ピッカーに表示するリアクション" | ||||
| reactionSettingDescription2: "ドラッグして並び替え、クリックして削除、+を押して追加します。" | ||||
| rememberNoteVisibility: "公開範囲を記憶する" | ||||
| @@ -193,7 +194,7 @@ clearQueueConfirmText: "未配達の投稿は配送されなくなります。 | ||||
| clearCachedFiles: "キャッシュをクリア" | ||||
| clearCachedFilesConfirm: "キャッシュされたリモートファイルをすべて削除しますか?" | ||||
| blockedInstances: "ブロックしたインスタンス" | ||||
| blockedInstancesDescription: "ブロックしたいインスタンスのホストを改行で区切って設定します。ブロックされたインスタンスは、このインスタンスとやり取りできなくなります。" | ||||
| blockedInstancesDescription: "ブロックしたいインスタンスのホストを改行で区切って設定します。ブロックされたインスタンスは、このインスタンスとやり取りできなくなります。サブドメインもブロックされます。" | ||||
| muteAndBlock: "ミュートとブロック" | ||||
| mutedUsers: "ミュートしたユーザー" | ||||
| blockedUsers: "ブロックしたユーザー" | ||||
| @@ -924,6 +925,293 @@ neverShow: "今後表示しない" | ||||
| remindMeLater: "また後で" | ||||
| didYouLikeMisskey: "Misskeyを気に入っていただけましたか?" | ||||
| pleaseDonate: "Misskeyは{host}が使用している無料のソフトウェアです。これからも開発を続けられるように、ぜひ寄付をお願いします!" | ||||
| roles: "ロール" | ||||
| role: "ロール" | ||||
| normalUser: "一般ユーザー" | ||||
| undefined: "未定義" | ||||
| assign: "アサイン" | ||||
| unassign: "アサインを解除" | ||||
| color: "色" | ||||
| manageCustomEmojis: "カスタム絵文字の管理" | ||||
| youCannotCreateAnymore: "これ以上作成することはできません。" | ||||
| cannotPerformTemporary: "一時的に利用できません" | ||||
| cannotPerformTemporaryDescription: "操作回数が制限を超過するため一時的に利用できません。しばらく時間を置いてから再度お試しください。" | ||||
| preset: "プリセット" | ||||
| selectFromPresets: "プリセットから選択" | ||||
| achievements: "実績" | ||||
|  | ||||
| _achievements: | ||||
|   earnedAt: "獲得日時" | ||||
|   _types: | ||||
|     _notes1: | ||||
|       title: "just setting up my msky" | ||||
|       description: "初めてノートを投稿した" | ||||
|       flavor: "良いMisskeyライフを!" | ||||
|     _notes10: | ||||
|       title: "いくつかのノート" | ||||
|       description: "ノートを10回投稿した" | ||||
|     _notes100: | ||||
|       title: "たくさんのノート" | ||||
|       description: "ノートを100回投稿した" | ||||
|     _notes500: | ||||
|       title: "ノートまみれ" | ||||
|       description: "ノートを500回投稿した" | ||||
|     _notes1000: | ||||
|       title: "ノートの山" | ||||
|       description: "ノートを1,000回投稿した" | ||||
|     _notes5000: | ||||
|       title: "湧き出るノート" | ||||
|       description: "ノートを5,000回投稿した" | ||||
|     _notes10000: | ||||
|       title: "スーパーノート" | ||||
|       description: "ノートを10,000回投稿した" | ||||
|     _notes20000: | ||||
|       title: "ニードモアノート" | ||||
|       description: "ノートを20,000回投稿した" | ||||
|     _notes30000: | ||||
|       title: "ノートノートノート" | ||||
|       description: "ノートを30,000回投稿した" | ||||
|     _notes40000: | ||||
|       title: "ノート工場" | ||||
|       description: "ノートを40,000回投稿した" | ||||
|     _notes50000: | ||||
|       title: "ノートの惑星" | ||||
|       description: "ノートを50,000回投稿した" | ||||
|     _notes60000: | ||||
|       title: "ノートクエーサー" | ||||
|       description: "ノートを60,000回投稿した" | ||||
|     _notes70000: | ||||
|       title: "ブラックノートホール" | ||||
|       description: "ノートを70,000回投稿した" | ||||
|     _notes80000: | ||||
|       title: "ノートギャラクシー" | ||||
|       description: "ノートを80,000回投稿した" | ||||
|     _notes90000: | ||||
|       title: "ノートバース" | ||||
|       description: "ノートを90,000回投稿した" | ||||
|     _notes100000: | ||||
|       title: "ALL YOUR NOTE ARE BELONG TO US" | ||||
|       description: "ノートを100,000回投稿した" | ||||
|       flavor: "そんなに書くことある?" | ||||
|     _login3: | ||||
|       title: "ビギナーⅠ" | ||||
|       description: "通算ログイン日数が3日" | ||||
|       flavor: "今日からね僕は ミスキストってことで" | ||||
|     _login7: | ||||
|       title: "ビギナーⅡ" | ||||
|       description: "通算ログイン日数が7日" | ||||
|       flavor: "慣れてきましたか?" | ||||
|     _login15: | ||||
|       title: "ビギナーⅢ" | ||||
|       description: "通算ログイン日数が15日" | ||||
|     _login30: | ||||
|       title: "ミスキストⅠ" | ||||
|       description: "通算ログイン日数が30日" | ||||
|     _login60: | ||||
|       title: "ミスキストⅡ" | ||||
|       description: "通算ログイン日数が60日" | ||||
|     _login100: | ||||
|       title: "ミスキストⅢ" | ||||
|       description: "通算ログイン日数が100日" | ||||
|       flavor: "そのユーザー、ミスキストにつき" | ||||
|     _login200: | ||||
|       title: "常連Ⅰ" | ||||
|       description: "通算ログイン日数が200日" | ||||
|     _login300: | ||||
|       title: "常連Ⅱ" | ||||
|       description: "通算ログイン日数が300日" | ||||
|     _login400: | ||||
|       title: "常連Ⅲ" | ||||
|       description: "通算ログイン日数が400日" | ||||
|     _login500: | ||||
|       title: "ベテランⅠ" | ||||
|       description: "通算ログイン日数が500日" | ||||
|       flavor: "諸君、私はノートが好きだ" | ||||
|     _login600: | ||||
|       title: "ベテランⅡ" | ||||
|       description: "通算ログイン日数が600日" | ||||
|     _login700: | ||||
|       title: "ベテランⅢ" | ||||
|       description: "通算ログイン日数が700日" | ||||
|     _login800: | ||||
|       title: "ノートマスターⅠ" | ||||
|       description: "通算ログイン日数が800日" | ||||
|     _login900: | ||||
|       title: "ノートマスターⅡ" | ||||
|       description: "通算ログイン日数が900日" | ||||
|     _login1000: | ||||
|       title: "ノートマスターⅢ" | ||||
|       description: "通算ログイン日数が1,000日" | ||||
|       flavor: "Misskeyを使ってくれてありがとう!" | ||||
|     _noteClipped1: | ||||
|       title: "クリップせずにはいられないな" | ||||
|       description: "初めてノートをクリップした" | ||||
|     _noteFavorited1: | ||||
|       title: "星をみるひと" | ||||
|       description: "初めてノートをお気に入りに登録した" | ||||
|     _profileFilled: | ||||
|       title: "準備万端" | ||||
|       description: "プロフィール設定を行った" | ||||
|     _markedAsCat: | ||||
|       title: "吾輩は猫である" | ||||
|       description: "アカウントをCatとして設定した" | ||||
|       flavor: "名前はまだない。" | ||||
|     _following1: | ||||
|       title: "はじめてのフォロー" | ||||
|       description: "初めてフォローした" | ||||
|     _following10: | ||||
|       title: "ついてく、ついてく" | ||||
|       description: "フォローが10人を超した" | ||||
|     _following50: | ||||
|       title: "友達たくさん" | ||||
|       description: "フォローが50人を超した" | ||||
|     _following100: | ||||
|       title: "友達100人" | ||||
|       description: "フォローが100人を超した" | ||||
|     _following300: | ||||
|       title: "友達過多" | ||||
|       description: "フォローが300人を超した" | ||||
|     _followers1: | ||||
|       title: "はじめてのフォロワー" | ||||
|       description: "初めてフォローされた" | ||||
|     _followers10: | ||||
|       title: "フォローミー!" | ||||
|       description: "フォロワーが10人を超した" | ||||
|     _followers50: | ||||
|       title: "ぞろぞろ" | ||||
|       description: "フォロワーが50人を超した" | ||||
|     _followers100: | ||||
|       title: "人気者" | ||||
|       description: "フォロワーが100人を超した" | ||||
|     _followers300: | ||||
|       title: "一列でお並びください" | ||||
|       description: "フォロワーが300人を超した" | ||||
|     _followers500: | ||||
|       title: "基地局" | ||||
|       description: "フォロワーが500人を超した" | ||||
|     _followers1000: | ||||
|       title: "インフルエンサー" | ||||
|       description: "フォロワーが1,000人を超した" | ||||
|     _collectAchievements30: | ||||
|       title: "実績コレクター" | ||||
|       description: "実績を30個以上獲得した" | ||||
|     _iLoveMisskey: | ||||
|       title: "I Love Misskey" | ||||
|       description: "\"I ❤ #Misskey\"を投稿した" | ||||
|       flavor: "Misskeyを使ってくださりありがとうございます! by 開発チーム" | ||||
|     _client30min: | ||||
|       title: "ひとやすみ" | ||||
|       description: "クライアントを起動してから30分以上経過した" | ||||
|     _noteDeletedWithin1min: | ||||
|       title: "いまのなし" | ||||
|       description: "投稿してから1分以内にその投稿を削除した" | ||||
|     _postedAtLateNight: | ||||
|       title: "夜行性" | ||||
|       description: "深夜にノートを投稿した" | ||||
|       flavor: "そろそろ寝よう。" | ||||
|     _postedAt0min0sec: | ||||
|       title: "時報" | ||||
|       description: "0分0秒にノートを投稿した" | ||||
|       flavor: "ポッ ポッ ポッ ピーン" | ||||
|     _selfQuote: | ||||
|       title: "自己言及" | ||||
|       description: "自分のノートを引用した" | ||||
|     _htl20npm: | ||||
|       title: "流れるTL" | ||||
|       description: "ホームタイムラインの流速が20npmを越す" | ||||
|     _driveFolderCircularReference: | ||||
|       title: "循環参照" | ||||
|       description: "ドライブのフォルダを再帰的な入れ子にしようとした" | ||||
|     _reactWithoutRead: | ||||
|       title: "ちゃんと読んだ?" | ||||
|       description: "100文字以上のテキストを含むノートに投稿されてから3秒以内にリアクションした" | ||||
|     _clickedClickHere: | ||||
|       title: "ここをクリック" | ||||
|       description: "ここをクリックした" | ||||
|     _justPlainLucky: | ||||
|       title: "単なるラッキー" | ||||
|       description: "10秒ごとに0.01%の確率で獲得" | ||||
|     _setNameToSyuilo: | ||||
|       title: "神様コンプレックス" | ||||
|       description: "名前を syuilo に設定した" | ||||
|     _passedSinceAccountCreated1: | ||||
|       title: "一周年" | ||||
|       description: "アカウント作成から1年経過した" | ||||
|     _passedSinceAccountCreated2: | ||||
|       title: "二周年" | ||||
|       description: "アカウント作成から2年経過した" | ||||
|     _passedSinceAccountCreated3: | ||||
|       title: "三周年" | ||||
|       description: "アカウント作成から3年経過した" | ||||
|     _loggedInOnBirthday: | ||||
|       title: "ハッピーバースデー" | ||||
|       description: "誕生日にログインした" | ||||
|     _cookieClicked: | ||||
|       title: "クッキーをクリックするゲーム" | ||||
|       description: "クッキーをクリックした" | ||||
|       flavor: "ソフト間違ってない?" | ||||
|     _brainDiver: | ||||
|       title: "Brain Diver" | ||||
|       description: "Brain Diverへのリンクを投稿した" | ||||
|       flavor: "Misskey-Misskey La-Tu-Ma" | ||||
|  | ||||
| _role: | ||||
|   new: "ロールの作成" | ||||
|   edit: "ロールの編集" | ||||
|   name: "ロール名" | ||||
|   description: "ロールの説明" | ||||
|   permission: "ロールの権限" | ||||
|   descriptionOfPermission: "<b>モデレーター</b>は基本的なモデレーションに関する操作を行えます。\n<b>管理者</b>はインスタンスの全ての設定を変更できます。" | ||||
|   assignTarget: "アサインターゲット" | ||||
|   descriptionOfAssignTarget: "<b>マニュアル</b>は誰がこのロールに含まれるかを手動で管理します。\n<b>コンディショナル</b>は条件を設定し、それに合致するユーザーが自動で含まれるようになります。" | ||||
|   manual: "マニュアル" | ||||
|   conditional: "コンディショナル" | ||||
|   condition: "条件" | ||||
|   isConditionalRole: "これはコンディショナルロールです。" | ||||
|   isPublic: "ロールを公開" | ||||
|   descriptionOfIsPublic: "ロールにアサインされたユーザーを誰でも見ることができます。また、ユーザーのプロフィールでこのロールが表示されます。" | ||||
|   options: "オプション" | ||||
|   policies: "ポリシー" | ||||
|   baseRole: "ベースロール" | ||||
|   useBaseValue: "ベースロールの値を使用" | ||||
|   chooseRoleToAssign: "アサインするロールを選択" | ||||
|   canEditMembersByModerator: "モデレーターのメンバー編集を許可" | ||||
|   descriptionOfCanEditMembersByModerator: "オンにすると、管理者に加えてモデレーターもこのロールへユーザーをアサイン/アサイン解除できるようになります。オフにすると管理者のみが行えます。" | ||||
|   priority: "優先度" | ||||
|   _priority: | ||||
|     low: "低" | ||||
|     middle: "中" | ||||
|     high: "高" | ||||
|   _options: | ||||
|     gtlAvailable: "グローバルタイムラインの閲覧" | ||||
|     ltlAvailable: "ローカルタイムラインの閲覧" | ||||
|     canPublicNote: "パブリック投稿の許可" | ||||
|     canInvite: "インスタンス招待コードの発行" | ||||
|     canManageCustomEmojis: "カスタム絵文字の管理" | ||||
|     driveCapacity: "ドライブ容量" | ||||
|     pinMax: "ノートのピン留めの最大数" | ||||
|     antennaMax: "アンテナの作成可能数" | ||||
|     wordMuteMax: "ワードミュートの最大文字数" | ||||
|     webhookMax: "Webhookの作成可能数" | ||||
|     clipMax: "クリップの作成可能数" | ||||
|     noteEachClipsMax: "クリップ内のノートの最大数" | ||||
|     userListMax: "ユーザーリストの作成可能数" | ||||
|     userEachUserListsMax: "ユーザーリスト内のユーザーの最大数" | ||||
|     rateLimitFactor: "レートリミット" | ||||
|     descriptionOfRateLimitFactor: "小さいほど制限が緩和され、大きいほど制限が強化されます。" | ||||
|     canHideAds: "広告の非表示" | ||||
|   _condition: | ||||
|     isLocal: "ローカルユーザー" | ||||
|     isRemote: "リモートユーザー" | ||||
|     createdLessThan: "アカウント作成から~以内" | ||||
|     createdMoreThan: "アカウント作成から~経過" | ||||
|     followersLessThanOrEq: "フォロワー数が~以下" | ||||
|     followersMoreThanOrEq: "フォロワー数が~以上" | ||||
|     followingLessThanOrEq: "フォロー数が~以下" | ||||
|     followingMoreThanOrEq: "フォロー数が~以上" | ||||
|     and: "~かつ~" | ||||
|     or: "~または~" | ||||
|     not: "~ではない" | ||||
|  | ||||
| _sensitiveMediaDetection: | ||||
|   description: "機械学習を使って自動でセンシティブなメディアを検出し、モデレーションに役立てることができます。サーバーの負荷が少し増えます。" | ||||
| @@ -962,6 +1250,7 @@ _accountDelete: | ||||
| _ad: | ||||
|   back: "戻る" | ||||
|   reduceFrequencyOfThisAd: "この広告の表示頻度を下げる" | ||||
|   hide: "表示しない" | ||||
|  | ||||
| _forgotPassword: | ||||
|   enterEmail: "アカウントに登録したメールアドレスを入力してください。そのアドレス宛てに、パスワードリセット用のリンクが送信されます。" | ||||
| @@ -1562,6 +1851,7 @@ _notification: | ||||
|   pollEnded: "アンケートの結果が出ました" | ||||
|   unreadAntennaNote: "アンテナ {name}" | ||||
|   emptyPushNotificationMessage: "プッシュ通知の更新をしました" | ||||
|   achievementEarned: "実績を獲得" | ||||
|  | ||||
|   _types: | ||||
|     all: "すべて" | ||||
|   | ||||
| @@ -915,8 +915,85 @@ caption: "キャプション" | ||||
| loggedInAsBot: "Botアカウントでログイン中やで" | ||||
| tools: "ツール" | ||||
| cannotLoad: "読み込めへんで" | ||||
| numberOfProfileView: "プロフィール表示回数" | ||||
| like: "ええやん!" | ||||
| unlike: "いいねを解除" | ||||
| numberOfLikes: "いいね数" | ||||
| show: "表示" | ||||
| neverShow: "今後表示しない" | ||||
| remindMeLater: "また後で" | ||||
| didYouLikeMisskey: "Misskeyを気に入っとっただけましたん?" | ||||
| pleaseDonate: "Misskeyは{host}が使用している無料のソフトウェアやで。これからも開発を続けれるように、寄付したってな~。" | ||||
| roles: "ロール" | ||||
| role: "ロール" | ||||
| normalUser: "一般ユーザー" | ||||
| undefined: "未定義" | ||||
| assign: "アサイン" | ||||
| unassign: "アサインを解除" | ||||
| color: "色" | ||||
| manageCustomEmojis: "カスタム絵文字の管理" | ||||
| youCannotCreateAnymore: "これ以上作れなさそうや" | ||||
| cannotPerformTemporary: "一時的に利用できへんで" | ||||
| cannotPerformTemporaryDescription: "操作回数が制限を超えたから一時的に利用できへんくなったで。ちょっと時間置いてからもう一回やってやー。" | ||||
| preset: "プリセット" | ||||
| selectFromPresets: "プリセットから選ぶ" | ||||
| _role: | ||||
|   new: "ロールの作成" | ||||
|   edit: "ロールの編集" | ||||
|   name: "ロール名" | ||||
|   description: "ロールの説明" | ||||
|   permission: "ロールの権限" | ||||
|   descriptionOfPermission: "<b>モデレーター</b>は基本的なモデレーションに関わる操作を行えるで。\n<b>管理者</b>はインスタンスの全ての設定を変更できるで。" | ||||
|   assignTarget: "アサインターゲット" | ||||
|   descriptionOfAssignTarget: "<b>マニュアル</b>は誰がこのロールに含まれてるかを手動で管理するで。\n<b>コンディショナル</b>は条件を設定して、それに合うユーザーが自動で含まれるようになるで。" | ||||
|   manual: "マニュアル" | ||||
|   conditional: "コンディショナル" | ||||
|   condition: "条件" | ||||
|   isConditionalRole: "これはコンディショナルロールやで" | ||||
|   isPublic: "ロールを公開" | ||||
|   descriptionOfIsPublic: "ロールにアサインされたユーザーを誰でも見ることができるで。そんで、ユーザーのプロフィールでこのロールが表示されるで。" | ||||
|   options: "オプション" | ||||
|   policies: "ポリシー" | ||||
|   baseRole: "ベースロール" | ||||
|   useBaseValue: "ベースロールの値を使用" | ||||
|   chooseRoleToAssign: "アサインするロールを選択" | ||||
|   canEditMembersByModerator: "モデレーターのメンバー編集を許可" | ||||
|   descriptionOfCanEditMembersByModerator: "オンにすると、管理者に加えてモデレーターもこのロールへユーザーをアサイン/アサイン解除できるようになるで。オフにすると管理者のみが行えるで。" | ||||
|   priority: "優先度" | ||||
|   _priority: | ||||
|     low: "低い" | ||||
|     middle: "中" | ||||
|     high: "高い" | ||||
|   _options: | ||||
|     gtlAvailable: "グローバルタイムラインの閲覧" | ||||
|     ltlAvailable: "ローカルタイムラインの閲覧" | ||||
|     canPublicNote: "パブリック投稿の許可" | ||||
|     canInvite: "インスタンス招待コードの発行" | ||||
|     canManageCustomEmojis: "カスタム絵文字の管理" | ||||
|     driveCapacity: "ドライブ容量" | ||||
|     pinMax: "ノートのピン留めの最大数" | ||||
|     antennaMax: "アンテナの作成可能数" | ||||
|     wordMuteMax: "ワードミュートの最大文字数" | ||||
|     webhookMax: "Webhockの作成可能数" | ||||
|     clipMax: "クリップの作成可能数" | ||||
|     noteEachClipsMax: "クリップ内のノートの最大数" | ||||
|     userListMax: "ユーザーリストの作成可能数" | ||||
|     userEachUserListsMax: "ユーザーリスト内のユーザーの最大数" | ||||
|     rateLimitFactor: "レートリミット" | ||||
|     descriptionOfRateLimitFactor: "ちっちゃいほど制限が緩くなって、大きいほど制限されるで。" | ||||
|     canHideAds: "広告を表示させへん" | ||||
|   _condition: | ||||
|     isLocal: "ローカルユーザー" | ||||
|     isRemote: "リモートユーザー" | ||||
|     createdLessThan: "アカウント作成から~以内" | ||||
|     createdMoreThan: "アカウント作成から~経過" | ||||
|     followersLessThanOrEq: "フォロワー数が~以下" | ||||
|     followersMoreThanOrEq: "フォロワー数が~以上" | ||||
|     followingLessThanOrEq: "フォロー数が~以下" | ||||
|     followingMoreThanOrEq: "フォロー数が~以上" | ||||
|     and: "~かつ~" | ||||
|     or: "~または~" | ||||
|     not: "~ではない" | ||||
| _sensitiveMediaDetection: | ||||
|   description: "機械学習を使って自動でセンシティブなメディアを検出して、モデレーションに役立てることができるで。サーバーの負荷が少し増えてまうなあ。" | ||||
|   sensitivity: "検出感度やで" | ||||
| @@ -949,6 +1026,7 @@ _accountDelete: | ||||
| _ad: | ||||
|   back: "戻る" | ||||
|   reduceFrequencyOfThisAd: "この広告の表示頻度を下げるで" | ||||
|   hide: "表示せん" | ||||
| _forgotPassword: | ||||
|   enterEmail: "アカウントに登録したメールアドレスをここに入力してや。そのアドレス宛に、パスワードリセット用のリンクが送られるから待っててな~。" | ||||
|   ifNoEmail: "メールアドレスを登録してへんのやったら、管理者まで教えてな~。" | ||||
| @@ -1318,10 +1396,12 @@ _widgets: | ||||
|   jobQueue: "ジョブキュー" | ||||
|   serverMetric: "サーバーメトリクス" | ||||
|   aiscript: "AiScriptコンソール" | ||||
|   aiscriptApp: "AiScript App" | ||||
|   aichan: "藍" | ||||
|   userList: "ユーザーリスト" | ||||
|   _userList: | ||||
|     chooseList: "リストを選ぶ" | ||||
|   clicker: "クリッカー" | ||||
| _cw: | ||||
|   hide: "隠す" | ||||
|   show: "続き見して!" | ||||
| @@ -1385,6 +1465,7 @@ _profile: | ||||
|   changeBanner: "バナー画像を変更するで" | ||||
| _exportOrImport: | ||||
|   allNotes: "全てのノート" | ||||
|   favoritedNotes: "お気に入りにしたノート" | ||||
|   followingList: "フォロー" | ||||
|   muteList: "ミュート" | ||||
|   blockingList: "ブロック" | ||||
| @@ -1423,7 +1504,16 @@ _timelines: | ||||
|   social: "ソーシャル" | ||||
|   global: "グローバル" | ||||
| _play: | ||||
|   new: "Playの作成" | ||||
|   edit: "Playの編集" | ||||
|   created: "Playを作ったで" | ||||
|   updated: "Playを更新したで" | ||||
|   deleted: "Playを消したで" | ||||
|   pageSetting: "Play設定" | ||||
|   editThisPage: "このPlayを編集" | ||||
|   viewSource: "ソースを表示" | ||||
|   my: "自分のPlay" | ||||
|   liked: "いいねしたPlay" | ||||
|   featured: "人気" | ||||
|   title: "タイトル" | ||||
|   script: "スクリプト" | ||||
|   | ||||
| @@ -15,7 +15,7 @@ gotIt: "알겠어요" | ||||
| cancel: "취소" | ||||
| noThankYou: "나중에" | ||||
| enterUsername: "유저명 입력" | ||||
| renotedBy: "{user}님의 리노트" | ||||
| renotedBy: "{user}님이 리노트" | ||||
| noNotes: "노트가 없습니다" | ||||
| noNotifications: "표시할 알림이 없습니다" | ||||
| instance: "인스턴스" | ||||
| @@ -907,7 +907,7 @@ subscribePushNotification: "푸시 알림 켜기" | ||||
| unsubscribePushNotification: "푸시 알림 끄기" | ||||
| pushNotificationAlreadySubscribed: "푸시 알림이 이미 켜져 있습니다" | ||||
| pushNotificationNotSupported: "브라우저나 인스턴스에서 푸시 알림이 지원되지 않습니다" | ||||
| sendPushNotificationReadMessage: "푸시 알림이니 메시지를 읽으면 푸시 알림을 삭제합니다" | ||||
| sendPushNotificationReadMessage: "푸시 알림이나 메시지를 읽은 뒤 푸시 알림을 삭제" | ||||
| sendPushNotificationReadMessageCaption: "「{emptyPushNotificationMessage}」이라는 알림이 잠깐 표시됩니다. 기기의 전력 소비량이 증가할 수 있습니다." | ||||
| windowMaximize: "최대화" | ||||
| windowRestore: "복구" | ||||
| @@ -924,6 +924,76 @@ neverShow: "다시 보지 않기" | ||||
| remindMeLater: "나중에 알림" | ||||
| didYouLikeMisskey: "Misskey가 마음에 드시나요?" | ||||
| pleaseDonate: "{host}은(는) 무료 소프트웨어 Misskey를 사용합니다. 후원을 통해 저희의 개발이 이어질 수 있게 도와주세요!" | ||||
| roles: "역할" | ||||
| role: "역할" | ||||
| normalUser: "일반 사용자" | ||||
| undefined: "정의되지 않음" | ||||
| assign: "할당" | ||||
| unassign: "할당 취소" | ||||
| color: "색" | ||||
| manageCustomEmojis: "커스텀 이모지 관리" | ||||
| youCannotCreateAnymore: "더 이상 생성할 수 없습니다." | ||||
| cannotPerformTemporary: "일시적으로 사용할 수 없음" | ||||
| cannotPerformTemporaryDescription: "조작 횟수 제한을 초과하여 일시적으로 사용이 불가합니다. 잠시 후 다시 시도해 주세요." | ||||
| preset: "프리셋" | ||||
| selectFromPresets: "프리셋에서 선택" | ||||
| _role: | ||||
|   new: "새 역할 생성" | ||||
|   edit: "역할 수정" | ||||
|   name: "역할 이름" | ||||
|   description: "역할 설명" | ||||
|   permission: "역할 권한" | ||||
|   descriptionOfPermission: "<b>모더레이터</b>는 기본적인 중재와 관련된 작업을 수행할 수 있습니다.\n<b>관리자</b>는 인스턴스의 모든 설정을 변경할 수 있습니다." | ||||
|   assignTarget: "할당 대상" | ||||
|   descriptionOfAssignTarget: "<b>수동</b>을 선택하면 누가 이 역할에 포함되는지를 수동으로 관리할 수 있습니다.\n<b>조건부</b>를 선택하면 조건을 설정해 일치하는 사용자를 자동으로 포함되게 할 수 있습니다." | ||||
|   manual: "수동" | ||||
|   conditional: "조건부" | ||||
|   condition: "조건" | ||||
|   isConditionalRole: "조건부 역할입니다." | ||||
|   isPublic: "역할 공개" | ||||
|   descriptionOfIsPublic: "역할에 할당된 사용자를 누구나 볼 수 있습니다. 또한 사용자 프로필에 이 역할이 표시됩니다." | ||||
|   options: "옵션" | ||||
|   policies: "정책" | ||||
|   baseRole: "기본 역할" | ||||
|   useBaseValue: "기본값 사용" | ||||
|   chooseRoleToAssign: "할당할 역할 선택" | ||||
|   canEditMembersByModerator: "모더레이터의 역할 수정 허용" | ||||
|   descriptionOfCanEditMembersByModerator: "이 옵션을 켜면 모더레이터도 이 역할에 사용자를 할당하거나 삭제할 수 있습니다. 꺼져 있으면 관리자만 할당이 가능합니다." | ||||
|   priority: "우선순위" | ||||
|   _priority: | ||||
|     low: "낮음" | ||||
|     middle: "보통" | ||||
|     high: "높음" | ||||
|   _options: | ||||
|     gtlAvailable: "글로벌 타임라인 보이기" | ||||
|     ltlAvailable: "로컬 타임라인 보이기" | ||||
|     canPublicNote: "공개 노트 허용" | ||||
|     canInvite: "인스턴스 초대 코드 발행" | ||||
|     canManageCustomEmojis: "커스텀 이모지 관리" | ||||
|     driveCapacity: "드라이브 용량" | ||||
|     pinMax: "고정할 수 있는 노트 수" | ||||
|     antennaMax: "최대 안테나 생성 허용 수" | ||||
|     wordMuteMax: "단어 뮤트할 수 있는 문자 수" | ||||
|     webhookMax: "생성할 수 있는 웹훅 수" | ||||
|     clipMax: "생성할 수 있는 클립 수" | ||||
|     noteEachClipsMax: "각 클립에 추가할 수 있는 노트 수" | ||||
|     userListMax: "생성할 수 있는 유저 리스트 수" | ||||
|     userEachUserListsMax: "유저 리스트당 최대 사용자 수" | ||||
|     rateLimitFactor: "속도 제한" | ||||
|     descriptionOfRateLimitFactor: "작을수록 제한이 완화되고, 클수록 제한이 강화됩니다." | ||||
|     canHideAds: "광고 숨기기" | ||||
|   _condition: | ||||
|     isLocal: "로컬 사용자" | ||||
|     isRemote: "리모트 사용자" | ||||
|     createdLessThan: "가압한 지 다음 일수 이내인 유저" | ||||
|     createdMoreThan: "가입한 지 다음 일수 이상인 유저" | ||||
|     followersLessThanOrEq: "팔로워 수가 다음 이하인 유저" | ||||
|     followersMoreThanOrEq: "팔로워 수가 다음 이상인 유저" | ||||
|     followingLessThanOrEq: "팔로잉 수가 다음 이하인 유저" | ||||
|     followingMoreThanOrEq: "팔로잉 수가 다음 이상인 유저" | ||||
|     and: "다음을 모두 만족" | ||||
|     or: "다음을 하나라도 만족" | ||||
|     not: "다음을 만족하지 않음" | ||||
| _sensitiveMediaDetection: | ||||
|   description: "기계학습을 통해 자동으로 민감한 미디어를 탐지하여, 모더레이션에 참고할 수 있도록 합니다. 서버의 부하를 약간 증가시킵니다." | ||||
|   sensitivity: "탐지 민감도" | ||||
| @@ -956,6 +1026,7 @@ _accountDelete: | ||||
| _ad: | ||||
|   back: "뒤로" | ||||
|   reduceFrequencyOfThisAd: "이 광고의 표시 빈도 낮추기" | ||||
|   hide: "보이지 않음" | ||||
| _forgotPassword: | ||||
|   enterEmail: "여기에 계정에 등록한 메일 주소를 입력해 주세요. 입력한 메일 주소로 비밀번호 재설정 링크를 발송합니다." | ||||
|   ifNoEmail: "메일 주소를 등록하지 않은 경우, 관리자에 문의해 주십시오." | ||||
| @@ -1327,7 +1398,7 @@ _widgets: | ||||
|   aiscript: "AiScript 콘솔" | ||||
|   aiscriptApp: "AiScript 앱" | ||||
|   aichan: "아이" | ||||
|   userList: "사용자 목록" | ||||
|   userList: "유저 리스트" | ||||
|   _userList: | ||||
|     chooseList: "리스트 선택" | ||||
|   clicker: "클리커" | ||||
|   | ||||
| @@ -868,6 +868,13 @@ sendPushNotificationReadMessageCaption: "Chwilowo pojawi się powiadomienie \"{e | ||||
| loggedInAsBot: "Jesteś obecnie zalogowany/a jako bot" | ||||
| like: "Polub" | ||||
| show: "Wyświetlanie" | ||||
| color: "Kolor" | ||||
| _role: | ||||
|   priority: "Priorytet" | ||||
|   _priority: | ||||
|     low: "Niski" | ||||
|     middle: "Średnie" | ||||
|     high: "Wysoki" | ||||
| _sensitiveMediaDetection: | ||||
|   description: "Zmniejsza wysiłek związany z moderacją serwera dzięki automatycznemu rozpoznawaniu zawartości NSFW za pomocą uczenia maszynowego. To nieznacznie zwiększy obciążenie serwera." | ||||
|   setSensitiveFlagAutomatically: "Oznacz jako NSFW" | ||||
| @@ -895,6 +902,7 @@ _accountDelete: | ||||
| _ad: | ||||
|   back: "Wróć" | ||||
|   reduceFrequencyOfThisAd: "Pokazuj tę reklamę rzadziej" | ||||
|   hide: "Nigdy nie pokazuj" | ||||
| _forgotPassword: | ||||
|   enterEmail: "Wpisz adres e-mail użyty do rejestracji. Zostanie do niego wysłany link, za pomocą którego możesz zresetować hasło." | ||||
|   ifNoEmail: "Jeżeli nie podano adresu e-mail podczas rejestracji, skontaktuj się z administratorem zamiast tego." | ||||
|   | ||||
| @@ -648,6 +648,9 @@ sent: "Trimite" | ||||
| searchByGoogle: "Caută" | ||||
| file: "Fișiere" | ||||
| show: "Arată" | ||||
| _role: | ||||
|   _priority: | ||||
|     middle: "Mediu" | ||||
| _email: | ||||
|   _follow: | ||||
|     title: "te-a urmărit" | ||||
|   | ||||
| @@ -866,6 +866,13 @@ windowMaximize: "Развернуть" | ||||
| windowRestore: "Восстановить" | ||||
| like: "Нравится!" | ||||
| show: "Отображение" | ||||
| color: "Цвет" | ||||
| _role: | ||||
|   priority: "Приоритет" | ||||
|   _priority: | ||||
|     low: "Низкий" | ||||
|     middle: "Средне" | ||||
|     high: "Высокий" | ||||
| _sensitiveMediaDetection: | ||||
|   description: "Машинное обучение может быть использовано для автоматического обнаружения чувствительных медиа для модерации. Нагрузка на сервер увеличивается незначительно." | ||||
|   setSensitiveFlagAutomatically: "Установить флаг NSFW" | ||||
| @@ -893,6 +900,7 @@ _accountDelete: | ||||
| _ad: | ||||
|   back: "Выход" | ||||
|   reduceFrequencyOfThisAd: "Реже показывать эту рекламу" | ||||
|   hide: "Не показывать" | ||||
| _forgotPassword: | ||||
|   enterEmail: "Введите адрес электронной почты, который ввели при регистрации. На неё будет выслана ссылка для смены пароля." | ||||
|   ifNoEmail: "Если вы не ввели свой адрес электронной почты, свяжитесь с администратором ресурса, чтобы сменить пароль." | ||||
|   | ||||
| @@ -917,6 +917,13 @@ neverShow: "Nabudúce nezobrazovať" | ||||
| remindMeLater: "Pripomenúť neskôr" | ||||
| didYouLikeMisskey: "Páči sa vám Misskey?" | ||||
| pleaseDonate: "Misskey je bezplatný softvér, ktorý používa {host}. Prosím, prispejte, aby sme ho mohli ďalej rozvíjať!" | ||||
| color: "Farba" | ||||
| _role: | ||||
|   priority: "Priorita" | ||||
|   _priority: | ||||
|     low: "Málo" | ||||
|     middle: "Stredné" | ||||
|     high: "Vysoká" | ||||
| _sensitiveMediaDetection: | ||||
|   description: "Strojové učenie sa použije na automatickú detekciu citlivých médií na účely ich moderovania. Mierne sa zvýši zaťaženie servera." | ||||
|   sensitivity: "Citlivosť detekcie" | ||||
| @@ -949,6 +956,7 @@ _accountDelete: | ||||
| _ad: | ||||
|   back: "Späť" | ||||
|   reduceFrequencyOfThisAd: "Túto reklamu zobrazovať menej" | ||||
|   hide: "Nikdy nezobrazovať" | ||||
| _forgotPassword: | ||||
|   enterEmail: "Zadajte emailovú adresu, ktorú ste použili pri registrácii. Pošleme vám na ňu odkaz, cez ktorý si môžete obnoviť heslo." | ||||
|   ifNoEmail: "Ak ste pri registrácii nepoužili email, prosím kontaktujte administrátora." | ||||
|   | ||||
| @@ -924,6 +924,60 @@ neverShow: "ไม่ต้องแสดงข้อความนี้อ | ||||
| remindMeLater: "ไว้ครั้งหน้าแล้วกัน" | ||||
| didYouLikeMisskey: "คุณเคยชอบ Misskey ไหม?" | ||||
| pleaseDonate: "{host} ใช้ซอฟต์แวร์ฟรี Misskey เราขอขอบคุณการบริจาคของคุณอย่างสูงเพื่อให้การพัฒนา Misskey สามารถดำเนินต่อไปได้นะ!" | ||||
| roles: "บทบาท" | ||||
| role: "บทบาท" | ||||
| normalUser: "ผู้ใช้มาตรฐาน" | ||||
| undefined: "ไม่ได้กำหนด" | ||||
| assign: "กำหนด" | ||||
| unassign: "ยังไม่มอบหมาย" | ||||
| color: "สี" | ||||
| manageCustomEmojis: "จัดการอีโมจิแบบกำหนดเอง" | ||||
| _role: | ||||
|   new: "บทบาทใหม่" | ||||
|   edit: "แก้ไขบทบาท" | ||||
|   name: "ชื่อบทบาท" | ||||
|   description: "คำอธิบายบทบาท" | ||||
|   permission: "สิทธิ์ตามบทบาท" | ||||
|   descriptionOfPermission: "<b>ผู้ดูแลกลั่นกรองเนื้อหา</b> สามารถดำเนินการดูแลขั้นพื้นฐานได้นะ\n<b>ผู้ดูแลระบบ</b> สามารถเปลี่ยนการตั้งค่าทั้งหมดของอินสแตนซ์ได้นะ" | ||||
|   assignTarget: "กำหนดเป้าหมาย" | ||||
|   descriptionOfAssignTarget: "<b>แมนนวล</b> เพื่อเปลี่ยนผู้ที่เป็นส่วนหนึ่งของบทบาทนี้และใครที่ไม่ใช่ด้วยตนเอง\n<b>เงื่อนไข</b> เพื่อให้ผู้ใช้ได้รับการกำหนดและนำออกจากบทบาทนี้โดยอัตโนมัติตามเงื่อนไขชุดหนึ่ง" | ||||
|   manual: "ปรับเอง" | ||||
|   conditional: "มีเงื่อนไข" | ||||
|   condition: "เงื่อนไข" | ||||
|   isConditionalRole: "นี่คือบทบาทที่มีเงื่อนไข" | ||||
|   isPublic: "บทบาทสาธารณะ" | ||||
|   descriptionOfIsPublic: "ทุกคนสามารถดูได้ว่าผู้ใช้งานนั้นได้รับมอบหมายบทบาทด้วยหรือไม่ \n\nบทบาทจะแสดงในโปรไฟล์ของผู้ใช้ด้วย" | ||||
|   options: "ตัวเลือกบทบาท" | ||||
|   baseRole: "บทบาทพื้นฐาน" | ||||
|   useBaseValue: "ใช้บทบาทพื้นฐานเริ่มต้น" | ||||
|   chooseRoleToAssign: "เลือกบทบาทที่ต้องการกำหนด" | ||||
|   canEditMembersByModerator: "อนุญาตให้ผู้ดูแลแก้ไขสมาชิก" | ||||
|   descriptionOfCanEditMembersByModerator: "เมื่อเปิดใช้ ผู้ดูแลนอกเหนือจากผู้ดูแลระบบแล้ว จะสามารถกำหนดและยกเลิกการมอบหมายบทบาทนี้ให้กับผู้ใช้ได้ เมื่อปิด เฉพาะผู้ดูแลระบบเท่านั้นที่จะสามารถกำหนดผู้ใช้ได้นะ" | ||||
|   priority: "ลำดับความสำคัญ" | ||||
|   _priority: | ||||
|     low: "ต่ำ" | ||||
|     middle: "ปานกลาง" | ||||
|     high: "สูง" | ||||
|   _options: | ||||
|     gtlAvailable: "การดูไทม์ไลน์ทั่วโลก" | ||||
|     ltlAvailable: "การดูไทม์ไลน์ในท้องถิ่น" | ||||
|     canPublicNote: "สามารถส่งโน้ตสาธารณะ" | ||||
|     canInvite: "สร้างรหัสเชิญอินสแตนซ์" | ||||
|     canManageCustomEmojis: "จัดการอีโมจิแบบกำหนดเอง" | ||||
|     driveCapacity: "ความจุของไดรฟ์" | ||||
|     antennaMax: "จำนวนสูงสุดของเสาอากาศ" | ||||
|   _condition: | ||||
|     isLocal: "ผู้ใช้ภายใน" | ||||
|     isRemote: "ผู้ใช้ระยะไกล" | ||||
|     createdLessThan: "สร้างน้อยกว่า" | ||||
|     createdMoreThan: "สร้างมากกว่า" | ||||
|     followersLessThanOrEq: "จำนวนผู้ติดตามน้อยกว่าหรือเท่ากับ\n" | ||||
|     followersMoreThanOrEq: "จำนวนผู้ติดตามมากกว่าหรือเท่ากับ\n" | ||||
|     followingLessThanOrEq: "จำนวนบัญชีต่อไปนี้คือ น้อยกว่าหรือเท่ากับ" | ||||
|     followingMoreThanOrEq: "จำนวนบัญชีต่อไปนี้คือ มากกว่าหรือเท่ากับ" | ||||
|     and: "และ" | ||||
|     or: "หรือ" | ||||
|     not: "ไม่" | ||||
| _sensitiveMediaDetection: | ||||
|   description: "ลดความพยายามในการดูแลเซิร์ฟเวอร์ผ่านการจดจำสื่อ NSFW โดยอัตโนมัติผ่านการเรียนรู้ของเครื่อง การทำสิ่งนี้อาจจะเพิ่มภาระบนเซิร์ฟเวอร์เล็กน้อย" | ||||
|   sensitivity: "การตรวจจับความไว" | ||||
| @@ -956,6 +1010,7 @@ _accountDelete: | ||||
| _ad: | ||||
|   back: "ย้อนกลับ" | ||||
|   reduceFrequencyOfThisAd: "แสดงโฆษณานี้ให้น้อยลง" | ||||
|   hide: "ไม่ต้องแสดง" | ||||
| _forgotPassword: | ||||
|   enterEmail: "ป้อนที่อยู่อีเมลที่คุณเคยใช้ในการลงทะเบียนไว้ ลิงก์ที่คุณสามารถรีเซ็ตรหัสผ่านได้นั้นจะถูกส่งไปนะ" | ||||
|   ifNoEmail: "ถ้าหากคุณไม่ได้ใช้อีเมลระหว่างการลงทะเบียน กรุณาติดต่อผู้ดูแลระบบอินสแตนซ์แทนนะ" | ||||
|   | ||||
| @@ -586,7 +586,7 @@ pluginTokenRequestedDescription: "Цей плагін зможе викорис | ||||
| notificationType: "Тип сповіщення" | ||||
| edit: "Редагувати" | ||||
| useStarForReactionFallback: "Використовувати ★ як запасний варіант, якщо емодзі реакції невідомий" | ||||
| emailServer: "Сервер електронної пошти" | ||||
| emailServer: "Email сервер" | ||||
| enableEmail: "Увімкнути функцію доставки пошти" | ||||
| emailConfigInfo: "Використовується для підтвердження електронної пошти підчас реєстрації, а також для відновлення паролю." | ||||
| email: "E-mail" | ||||
| @@ -892,8 +892,18 @@ unsubscribePushNotification: "Вимкнути push-сповіщення" | ||||
| windowMaximize: "Розгорнути" | ||||
| windowRestore: "Відновити" | ||||
| caption: "Підпис" | ||||
| tools: "Інструменти" | ||||
| like: "Вподобати" | ||||
| unlike: "Не вподобати" | ||||
| numberOfLikes: "Вподобання" | ||||
| show: "Відображення" | ||||
| color: "Колір" | ||||
| _role: | ||||
|   priority: "Пріоритет" | ||||
|   _priority: | ||||
|     low: "Низький" | ||||
|     middle: "Середній" | ||||
|     high: "Високий" | ||||
| _sensitiveMediaDetection: | ||||
|   sensitivity: "Чутливість детектування" | ||||
|   setSensitiveFlagAutomatically: "Позначити як NSFW" | ||||
| @@ -919,6 +929,7 @@ _accountDelete: | ||||
| _ad: | ||||
|   back: "Назад" | ||||
|   reduceFrequencyOfThisAd: "Показувати цю рекламу менше" | ||||
|   hide: "Не відображати" | ||||
| _gallery: | ||||
|   my: "Моя галерея" | ||||
|   liked: "Вподобане" | ||||
|   | ||||
| @@ -896,6 +896,13 @@ account: "Tài khoản của bạn" | ||||
| move: "Di chuyển" | ||||
| like: "Thích" | ||||
| show: "Hiển thị" | ||||
| color: "Màu sắc" | ||||
| _role: | ||||
|   priority: "Ưu tiên" | ||||
|   _priority: | ||||
|     low: "Thấp" | ||||
|     middle: "Vừa" | ||||
|     high: "Cao" | ||||
| _sensitiveMediaDetection: | ||||
|   description: "Giảm nỗ lực kiểm duyệt máy chủ thông qua việc tự động nhận dạng media NSFW thông qua học máy. Điều này sẽ làm tăng một chút áp lực trên máy chủ." | ||||
|   sensitivity: "Phát hiện nhạy cảm" | ||||
| @@ -928,6 +935,7 @@ _accountDelete: | ||||
| _ad: | ||||
|   back: "Quay lại" | ||||
|   reduceFrequencyOfThisAd: "Hiện ít lại" | ||||
|   hide: "Không hiển thị" | ||||
| _forgotPassword: | ||||
|   enterEmail: "Nhập địa chỉ email bạn đã sử dụng để đăng ký. Một liên kết mà bạn có thể đặt lại mật khẩu của mình sau đó sẽ được gửi đến nó." | ||||
|   ifNoEmail: "Nếu bạn không sử dụng email lúc đăng ký, vui lòng liên hệ với quản trị viên." | ||||
|   | ||||
| @@ -13,7 +13,7 @@ fetchingAsApObject: "在联邦宇宙查询中..." | ||||
| ok: "OK" | ||||
| gotIt: "我明白了" | ||||
| cancel: "取消" | ||||
| noThankYou: "不用" | ||||
| noThankYou: "不用,谢谢" | ||||
| enterUsername: "输入用户名" | ||||
| renotedBy: "由 {user} 转贴" | ||||
| noNotes: "没有帖子" | ||||
| @@ -607,7 +607,7 @@ wordMute: "文字屏蔽" | ||||
| regexpError: "正则表达式错误" | ||||
| regexpErrorDescription: "{tab} 屏蔽文字的第 {line} 行的正则表达式有错误:" | ||||
| instanceMute: "实例的屏蔽" | ||||
| userSaysSomething: "{name}说了什么,但是被您屏蔽了" | ||||
| userSaysSomething: "{name}说了什么,但是被屏蔽词过滤了" | ||||
| makeActive: "启用" | ||||
| display: "显示" | ||||
| copy: "复制" | ||||
| @@ -826,7 +826,7 @@ makeReactionsPublicDescription: "将您发表过的回应设置成公开可见 | ||||
| classic: "经典" | ||||
| muteThread: "屏蔽帖子列表" | ||||
| unmuteThread: "取消屏蔽帖子列表" | ||||
| ffVisibility: "连接的可见范围" | ||||
| ffVisibility: "关注关系的可见范围" | ||||
| ffVisibilityDescription: "您可以设置您的关注/关注者信息的公开范围" | ||||
| continueThread: "查看更多帖子" | ||||
| deleteAccountConfirm: "将要删除账户。是否确认?" | ||||
| @@ -924,6 +924,76 @@ neverShow: "不再显示" | ||||
| remindMeLater: "稍后提醒我" | ||||
| didYouLikeMisskey: "您喜欢Misskey吗?" | ||||
| pleaseDonate: "Misskey是{host}所使用的免费软件。为了今后也能够维持Misskey的开发,请在有余力的情况下进行捐助!" | ||||
| roles: "角色" | ||||
| role: "角色" | ||||
| normalUser: "普通用户" | ||||
| undefined: "未定义" | ||||
| assign: "分配" | ||||
| unassign: "取消分配" | ||||
| color: "颜色" | ||||
| manageCustomEmojis: "管理自定义表情符号" | ||||
| youCannotCreateAnymore: "抱歉,您无法再创建更多了。" | ||||
| cannotPerformTemporary: "暂时不可用" | ||||
| cannotPerformTemporaryDescription: "因操作过于频繁,暂时不可用,请稍后再试。" | ||||
| preset: "預設值" | ||||
| selectFromPresets: "從預設值中選擇" | ||||
| _role: | ||||
|   new: "创建角色" | ||||
|   edit: "编辑角色" | ||||
|   name: "角色名称" | ||||
|   description: "角色描述" | ||||
|   permission: "角色权限" | ||||
|   descriptionOfPermission: "<b>监察员</b>可以执行基本的审核操作。\n<b>管理员</b>可以更改实例的所有设置。" | ||||
|   assignTarget: "授权对象" | ||||
|   descriptionOfAssignTarget: "<b>手动</b>指手动选择谁被包括在这个角色中。\n<b>符合条件</b>指设置条件以自动包括符合条件的用户。" | ||||
|   manual: "手动" | ||||
|   conditional: "符合条件" | ||||
|   condition: "条件" | ||||
|   isConditionalRole: "这是一个条件控制的角色。" | ||||
|   isPublic: "角色公开" | ||||
|   descriptionOfIsPublic: "任何人都可以看到分配该角色的用户。而用户的个人资料也将显示该角色。" | ||||
|   options: "选项" | ||||
|   policies: "策略" | ||||
|   baseRole: "基本角色" | ||||
|   useBaseValue: "使用基本角色的值" | ||||
|   chooseRoleToAssign: "选择要分配的角色" | ||||
|   canEditMembersByModerator: "允许监察者编辑成员" | ||||
|   descriptionOfCanEditMembersByModerator: "如果选中,监察者和管理员都能够为用户分配/取消分配角色。如果未选中,则只有管理员可以执行此操作。" | ||||
|   priority: "优先级" | ||||
|   _priority: | ||||
|     low: "低" | ||||
|     middle: "中" | ||||
|     high: "高" | ||||
|   _options: | ||||
|     gtlAvailable: "查看全局时间线" | ||||
|     ltlAvailable: "查看本地时间线" | ||||
|     canPublicNote: "允许公开发帖" | ||||
|     canInvite: "发放实例邀请码" | ||||
|     canManageCustomEmojis: "管理自定义表情符号" | ||||
|     driveCapacity: "网盘容量" | ||||
|     pinMax: "帖子置顶数量限制" | ||||
|     antennaMax: "可创建的最大天线数量" | ||||
|     wordMuteMax: "屏蔽词的字数限制" | ||||
|     webhookMax: "Webhook 创建数量限制" | ||||
|     clipMax: "便签创建数量限制" | ||||
|     noteEachClipsMax: "单个便签内的贴文数量限制" | ||||
|     userListMax: "用户列表创建数量限制" | ||||
|     userEachUserListsMax: "单个用户列表内用户数量限制" | ||||
|     rateLimitFactor: "速率限制" | ||||
|     descriptionOfRateLimitFactor: "值越小限制越少,值越大限制越多。" | ||||
|     canHideAds: "可以隐藏广告" | ||||
|   _condition: | ||||
|     isLocal: "是本地用户" | ||||
|     isRemote: "是远程用户" | ||||
|     createdLessThan: "账户创建时间少于" | ||||
|     createdMoreThan: "账户创建时间超过" | ||||
|     followersLessThanOrEq: "关注者不多于" | ||||
|     followersMoreThanOrEq: "关注者不少于" | ||||
|     followingLessThanOrEq: "关注中不多于" | ||||
|     followingMoreThanOrEq: "关注中不少于" | ||||
|     and: "符合以下全部条件" | ||||
|     or: "符合以下任一条件" | ||||
|     not: "不符合以下任何条件" | ||||
| _sensitiveMediaDetection: | ||||
|   description: "可以使用机器学习技术自动检测敏感媒体,以便进行审核。服务器负载将略微增加。" | ||||
|   sensitivity: "检测敏感度" | ||||
| @@ -939,7 +1009,7 @@ _emailUnavailable: | ||||
|   mx: "邮件服务器不正确" | ||||
|   smtp: "邮件服务器没有响应" | ||||
| _ffVisibility: | ||||
|   public: "发布" | ||||
|   public: "公开" | ||||
|   followers: "只有关注你的用户能看到" | ||||
|   private: "私密" | ||||
| _signup: | ||||
| @@ -956,6 +1026,7 @@ _accountDelete: | ||||
| _ad: | ||||
|   back: "返回" | ||||
|   reduceFrequencyOfThisAd: "减少此广告的频率" | ||||
|   hide: "不显示" | ||||
| _forgotPassword: | ||||
|   enterEmail: "请输入您验证账号时用的电子邮箱地址,密码重置链接将发送至该邮箱上。" | ||||
|   ifNoEmail: "如果您没有使用电子邮件地址进行验证,请联系管理员。" | ||||
|   | ||||
| @@ -324,8 +324,8 @@ integration: "整合" | ||||
| connectService: "己連結" | ||||
| disconnectService: "己斷開 " | ||||
| enableLocalTimeline: "開啟本地時間軸" | ||||
| enableGlobalTimeline: "啟用公開時間軸" | ||||
| disablingTimelinesInfo: "即使您關閉了時間線功能,管理員和協調人仍可以繼續使用,以方便您。" | ||||
| enableGlobalTimeline: "啟用全域時間軸" | ||||
| disablingTimelinesInfo: "為了方便,即使您關閉了時間線功能,管理員和審核員仍可以繼續使用。" | ||||
| registration: "註冊" | ||||
| enableRegistration: "開啟新使用者註冊" | ||||
| invite: "邀請" | ||||
| @@ -388,8 +388,8 @@ aboutMisskey: "關於 Misskey" | ||||
| administrator: "管理員" | ||||
| token: "權杖" | ||||
| twoStepAuthentication: "兩階段驗證" | ||||
| moderator: "板主" | ||||
| moderation: "言論調節" | ||||
| moderator: "監察員" | ||||
| moderation: "監察" | ||||
| nUsersMentioned: "提到了{n}" | ||||
| securityKey: "安全金鑰" | ||||
| securityKeyName: "金鑰名稱" | ||||
| @@ -869,7 +869,7 @@ recommended: "推薦" | ||||
| check: "檢查" | ||||
| driveCapOverrideLabel: "更改這個使用者的雲端硬碟容量上限" | ||||
| driveCapOverrideCaption: "如果指定0以下的值,就會被取消。" | ||||
| requireAdminForView: "必須以管理者帳號登入才可以檢視。" | ||||
| requireAdminForView: "必須以管理員帳號登入才可以檢視。" | ||||
| isSystemAccount: "由系統自動建立與管理的帳號。" | ||||
| typeToConfirm: "要執行這項操作,請輸入 {x} " | ||||
| deleteAccount: "刪除帳號" | ||||
| @@ -924,6 +924,76 @@ neverShow: "不再顯示" | ||||
| remindMeLater: "以後再說" | ||||
| didYouLikeMisskey: "您是否喜愛Misskey呢?" | ||||
| pleaseDonate: "Misskey是由{host}使用的免費軟體。請贊助我們,讓開發能夠持續!" | ||||
| roles: "角色" | ||||
| role: "角色" | ||||
| normalUser: "一般使用者" | ||||
| undefined: "未定義" | ||||
| assign: "指派" | ||||
| unassign: "取消指派" | ||||
| color: "顏色" | ||||
| manageCustomEmojis: "管理自訂表情符號" | ||||
| youCannotCreateAnymore: "您無法再建立更多了。" | ||||
| cannotPerformTemporary: "暫時無法進行" | ||||
| cannotPerformTemporaryDescription: "由於超過操作次數限制,暫時無法進行。請過一段時間之後再嘗試。" | ||||
| preset: "預設值" | ||||
| selectFromPresets: "從預設值中選擇" | ||||
| _role: | ||||
|   new: "建立角色" | ||||
|   edit: "編輯角色" | ||||
|   name: "角色名稱" | ||||
|   description: "角色描述 " | ||||
|   permission: "角色的權限" | ||||
|   descriptionOfPermission: "<b>審核員</b>執行與審核相關的基本操作。\n<b>管理員</b>能變更實例的全部設定。" | ||||
|   assignTarget: "指派目標" | ||||
|   descriptionOfAssignTarget: "<b>手動</b>是以手動管理這個角色包含的人員。\n<b>符合條件</b>是設定條件以自動包含符合條件的使用者。" | ||||
|   manual: "手動" | ||||
|   conditional: "符合條件" | ||||
|   condition: "條件" | ||||
|   isConditionalRole: "這是條件角色。" | ||||
|   isPublic: "角色為公開" | ||||
|   descriptionOfIsPublic: "任何人都可以看到被指派了角色的使用者。此外,使用者的個人檔案將顯示這個角色。" | ||||
|   options: "選項" | ||||
|   policies: "政策" | ||||
|   baseRole: "基本角色" | ||||
|   useBaseValue: "使用基本角色的值" | ||||
|   chooseRoleToAssign: "選擇要指派的角色" | ||||
|   canEditMembersByModerator: "允許編輯監察員的成員" | ||||
|   descriptionOfCanEditMembersByModerator: "如果開啟,管理員與監察員都可以為使用者指派/解除指派該角色。如果關閉,則只有管理員可以執行。" | ||||
|   priority: "優先級" | ||||
|   _priority: | ||||
|     low: "低" | ||||
|     middle: "中" | ||||
|     high: "高" | ||||
|   _options: | ||||
|     gtlAvailable: "瀏覽全域時間軸" | ||||
|     ltlAvailable: "瀏覽本地時間軸" | ||||
|     canPublicNote: "允許公開貼文" | ||||
|     canInvite: "發行實例邀請碼" | ||||
|     canManageCustomEmojis: "管理自訂表情符號" | ||||
|     driveCapacity: "雲端硬碟容量" | ||||
|     pinMax: "置頂貼文的最大數量" | ||||
|     antennaMax: "可建立的天線數量" | ||||
|     wordMuteMax: "靜音文字的最大字數" | ||||
|     webhookMax: "可建立的Webhook數量" | ||||
|     clipMax: "可建立的摘錄數量" | ||||
|     noteEachClipsMax: "摘錄內貼文的最大數量" | ||||
|     userListMax: "可建立的使用者清單數量" | ||||
|     userEachUserListsMax: "使用者清單內使用者的最大數量" | ||||
|     rateLimitFactor: "速率限制" | ||||
|     descriptionOfRateLimitFactor: "值越小限制越少,值越大限制越多。" | ||||
|     canHideAds: "不顯示廣告" | ||||
|   _condition: | ||||
|     isLocal: "本地使用者" | ||||
|     isRemote: "遠端使用者" | ||||
|     createdLessThan: "自建立帳戶開始~以內" | ||||
|     createdMoreThan: "自建立帳戶開始~經過" | ||||
|     followersLessThanOrEq: "追隨者人數在~以下" | ||||
|     followersMoreThanOrEq: "追隨者人數在~以上" | ||||
|     followingLessThanOrEq: "追隨人數在~以下" | ||||
|     followingMoreThanOrEq: "追隨人數在~以上" | ||||
|     and: "~和~" | ||||
|     or: "~或~" | ||||
|     not: "~否" | ||||
| _sensitiveMediaDetection: | ||||
|   description: "您可以使用機器學習自動檢測敏感媒體並將其用於審核。 伺服器的負荷會稍微增加。" | ||||
|   sensitivity: "檢測敏感度" | ||||
| @@ -956,6 +1026,7 @@ _accountDelete: | ||||
| _ad: | ||||
|   back: "返回" | ||||
|   reduceFrequencyOfThisAd: "降低此廣告的頻率 " | ||||
|   hide: "隱藏" | ||||
| _forgotPassword: | ||||
|   enterEmail: "請輸入您的帳戶註冊的電子郵件地址。 密碼重置連結將被發送到該電子郵件地址。" | ||||
|   ifNoEmail: "如果您還沒有註冊您的電子郵件地址,請聯繫管理員。 " | ||||
|   | ||||
							
								
								
									
										51
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										51
									
								
								package.json
									
									
									
									
									
								
							| @@ -1,12 +1,12 @@ | ||||
| { | ||||
| 	"name": "misskey", | ||||
| 	"version": "13.0.0-beta.39", | ||||
| 	"codename": "indigo", | ||||
| 	"version": "13.0.0", | ||||
| 	"codename": "nasubi", | ||||
| 	"repository": { | ||||
| 		"type": "git", | ||||
| 		"url": "https://github.com/misskey-dev/misskey.git" | ||||
| 	}, | ||||
| 	"packageManager": "yarn@3.3.0", | ||||
| 	"packageManager": "pnpm@7.24.3", | ||||
| 	"workspaces": [ | ||||
| 		"packages/frontend", | ||||
| 		"packages/backend", | ||||
| @@ -15,27 +15,27 @@ | ||||
| 	"private": true, | ||||
| 	"scripts": { | ||||
| 		"build-pre": "node ./scripts/build-pre.js", | ||||
| 		"build": "yarn build-pre && yarn workspaces foreach run build && yarn run gulp", | ||||
| 		"build": "pnpm build-pre && pnpm -r build && pnpm gulp", | ||||
| 		"start": "cd packages/backend && node ./built/boot/index.js", | ||||
| 		"start:test": "cd packages/backend && cross-env NODE_ENV=test node ./built/boot/index.js", | ||||
| 		"init": "yarn migrate", | ||||
| 		"migrate": "cd packages/backend && yarn run typeorm migration:run -d ormconfig.js", | ||||
| 		"migrateandstart": "yarn migrate && yarn start", | ||||
| 		"gulp": "gulp build", | ||||
| 		"watch": "yarn dev", | ||||
| 		"init": "pnpm migrate", | ||||
| 		"migrate": "cd packages/backend && pnpm typeorm migration:run -d ormconfig.js", | ||||
| 		"migrateandstart": "pnpm migrate && pnpm start", | ||||
| 		"gulp": "pnpm exec gulp build", | ||||
| 		"watch": "pnpm dev", | ||||
| 		"dev": "node ./scripts/dev.js", | ||||
| 		"lint": "yarn workspaces foreach run lint", | ||||
| 		"cy:open": "cypress open --browser --e2e --config-file=cypress.config.ts", | ||||
| 		"cy:run": "cypress run", | ||||
| 		"e2e": "start-server-and-test start:test http://localhost:61812 cy:run", | ||||
| 		"jest": "cd packages/backend && cross-env NODE_ENV=test node --experimental-vm-modules --experimental-import-meta-resolve node_modules/jest/bin/jest.js --forceExit --runInBand", | ||||
| 		"jest-and-coverage": "cd packages/backend && cross-env NODE_ENV=test node --experimental-vm-modules --experimental-import-meta-resolve node_modules/jest/bin/jest.js --coverage --forceExit --runInBand", | ||||
| 		"test": "yarn jest", | ||||
| 		"test-and-coverage": "yarn jest-and-coverage", | ||||
| 		"format": "gulp format", | ||||
| 		"lint": "pnpm -r lint", | ||||
| 		"cy:open": "pnpm cypress open --browser --e2e --config-file=cypress.config.ts", | ||||
| 		"cy:run": "pnpm cypress run", | ||||
| 		"e2e": "pnpm start-server-and-test start:test http://localhost:61812 cy:run", | ||||
| 		"jest": "cd packages/backend && pnpm cross-env NODE_ENV=test node --experimental-vm-modules --experimental-import-meta-resolve node_modules/jest/bin/jest.js --forceExit --runInBand", | ||||
| 		"jest-and-coverage": "cd packages/backend && pnpm cross-env NODE_ENV=test node --experimental-vm-modules --experimental-import-meta-resolve node_modules/jest/bin/jest.js --coverage --forceExit --runInBand", | ||||
| 		"test": "pnpm jest", | ||||
| 		"test-and-coverage": "pnpm jest-and-coverage", | ||||
| 		"format": "pnpm exec gulp format", | ||||
| 		"clean": "node ./scripts/clean.js", | ||||
| 		"clean-all": "node ./scripts/clean-all.js", | ||||
| 		"cleanall": "yarn clean-all" | ||||
| 		"cleanall": "pnpm clean-all" | ||||
| 	}, | ||||
| 	"resolutions": { | ||||
| 		"chokidar": "^3.3.1", | ||||
| @@ -48,17 +48,20 @@ | ||||
| 		"gulp-rename": "2.0.0", | ||||
| 		"gulp-replace": "1.1.4", | ||||
| 		"gulp-terser": "2.1.0", | ||||
| 		"js-yaml": "4.1.0" | ||||
| 		"js-yaml": "4.1.0", | ||||
| 		"typescript": "4.9.4" | ||||
| 	}, | ||||
| 	"devDependencies": { | ||||
| 		"@types/gulp": "4.0.10", | ||||
| 		"@types/gulp-rename": "2.0.1", | ||||
| 		"@typescript-eslint/eslint-plugin": "5.48.0", | ||||
| 		"@typescript-eslint/parser": "5.48.0", | ||||
| 		"@typescript-eslint/eslint-plugin": "5.48.1", | ||||
| 		"@typescript-eslint/parser": "5.48.1", | ||||
| 		"cross-env": "7.0.3", | ||||
| 		"cypress": "12.3.0", | ||||
| 		"eslint": "^8.31.0", | ||||
| 		"start-server-and-test": "1.15.2", | ||||
| 		"typescript": "4.9.4" | ||||
| 		"start-server-and-test": "1.15.2" | ||||
| 	}, | ||||
| 	"optionalDependencies": { | ||||
| 		"@tensorflow/tfjs-core": "^4.2.0" | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -9,7 +9,17 @@ | ||||
|     "transform": { | ||||
|       "legacyDecorator": true, | ||||
|       "decoratorMetadata": true | ||||
|     } | ||||
|     }, | ||||
| 		"experimental": { | ||||
| 			"keepImportAssertions": true | ||||
| 		}, | ||||
| 		"baseUrl": ".", | ||||
| 		"paths": { | ||||
| 			"@/*": [ | ||||
| 				"./src/*" | ||||
| 			] | ||||
| 		}, | ||||
| 		"target": "es2021" | ||||
|   }, | ||||
|   "minify": false | ||||
| } | ||||
|   | ||||
							
								
								
									
										37
									
								
								packages/backend/migration/1673500412259-Role.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								packages/backend/migration/1673500412259-Role.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,37 @@ | ||||
| export class Role1673500412259 { | ||||
|     name = 'Role1673500412259' | ||||
|  | ||||
|     async up(queryRunner) { | ||||
|         await queryRunner.query(`CREATE TABLE "role" ("id" character varying(32) NOT NULL, "createdAt" TIMESTAMP WITH TIME ZONE NOT NULL, "updatedAt" TIMESTAMP WITH TIME ZONE NOT NULL, "name" character varying(256) NOT NULL, "description" character varying(1024) NOT NULL, "isPublic" boolean NOT NULL DEFAULT false, "isModerator" boolean NOT NULL DEFAULT false, "isAdministrator" boolean NOT NULL DEFAULT false, "options" jsonb NOT NULL DEFAULT '{}', CONSTRAINT "PK_b36bcfe02fc8de3c57a8b2391c2" PRIMARY KEY ("id")); COMMENT ON COLUMN "role"."createdAt" IS 'The created date of the Role.'; COMMENT ON COLUMN "role"."updatedAt" IS 'The updated date of the Role.'`); | ||||
|         await queryRunner.query(`CREATE TABLE "role_assignment" ("id" character varying(32) NOT NULL, "createdAt" TIMESTAMP WITH TIME ZONE NOT NULL, "userId" character varying(32) NOT NULL, "roleId" character varying(32) NOT NULL, CONSTRAINT "PK_7e79671a8a5db18936173148cb4" PRIMARY KEY ("id")); COMMENT ON COLUMN "role_assignment"."createdAt" IS 'The created date of the RoleAssignment.'; COMMENT ON COLUMN "role_assignment"."userId" IS 'The user ID.'; COMMENT ON COLUMN "role_assignment"."roleId" IS 'The role ID.'`); | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_db5b72c16227c97ca88734d5c2" ON "role_assignment" ("userId") `); | ||||
|         await queryRunner.query(`CREATE INDEX "IDX_f0de67fd09cd3cd0aabca79994" ON "role_assignment" ("roleId") `); | ||||
|         await queryRunner.query(`CREATE UNIQUE INDEX "IDX_0953deda7ce6e1448e935859e5" ON "role_assignment" ("userId", "roleId") `); | ||||
| 				await queryRunner.query(`ALTER TABLE "user" RENAME COLUMN "isAdmin" TO "isRoot"`); | ||||
| 				await queryRunner.query(`ALTER TABLE "user" DROP COLUMN "isModerator"`); | ||||
|         await queryRunner.query(`ALTER TABLE "user" DROP COLUMN "driveCapacityOverrideMb"`); | ||||
|         await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "disableLocalTimeline"`); | ||||
|         await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "disableGlobalTimeline"`); | ||||
|         await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "localDriveCapacityMb"`); | ||||
|         await queryRunner.query(`ALTER TABLE "meta" ADD "defaultRoleOverride" jsonb NOT NULL DEFAULT '{}'`); | ||||
|         await queryRunner.query(`ALTER TABLE "role_assignment" ADD CONSTRAINT "FK_db5b72c16227c97ca88734d5c2b" FOREIGN KEY ("userId") REFERENCES "user"("id") ON DELETE CASCADE ON UPDATE NO ACTION`); | ||||
|         await queryRunner.query(`ALTER TABLE "role_assignment" ADD CONSTRAINT "FK_f0de67fd09cd3cd0aabca79994d" FOREIGN KEY ("roleId") REFERENCES "role"("id") ON DELETE CASCADE ON UPDATE NO ACTION`); | ||||
|     } | ||||
|  | ||||
|     async down(queryRunner) { | ||||
|         await queryRunner.query(`ALTER TABLE "role_assignment" DROP CONSTRAINT "FK_f0de67fd09cd3cd0aabca79994d"`); | ||||
|         await queryRunner.query(`ALTER TABLE "role_assignment" DROP CONSTRAINT "FK_db5b72c16227c97ca88734d5c2b"`); | ||||
|         await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "defaultRoleOverride"`); | ||||
|         await queryRunner.query(`ALTER TABLE "meta" ADD "localDriveCapacityMb" integer NOT NULL DEFAULT '1024'`); | ||||
|         await queryRunner.query(`ALTER TABLE "meta" ADD "disableGlobalTimeline" boolean NOT NULL DEFAULT false`); | ||||
|         await queryRunner.query(`ALTER TABLE "meta" ADD "disableLocalTimeline" boolean NOT NULL DEFAULT false`); | ||||
|         await queryRunner.query(`ALTER TABLE "user" ADD "driveCapacityOverrideMb" integer`); | ||||
| 				await queryRunner.query(`ALTER TABLE "user" ADD "isModerator" boolean NOT NULL DEFAULT false`); | ||||
| 				await queryRunner.query(`ALTER TABLE "user" RENAME COLUMN "isRoot" TO "isAdmin"`); | ||||
|         await queryRunner.query(`DROP INDEX "public"."IDX_0953deda7ce6e1448e935859e5"`); | ||||
|         await queryRunner.query(`DROP INDEX "public"."IDX_f0de67fd09cd3cd0aabca79994"`); | ||||
|         await queryRunner.query(`DROP INDEX "public"."IDX_db5b72c16227c97ca88734d5c2"`); | ||||
|         await queryRunner.query(`DROP TABLE "role_assignment"`); | ||||
|         await queryRunner.query(`DROP TABLE "role"`); | ||||
|     } | ||||
| } | ||||
							
								
								
									
										11
									
								
								packages/backend/migration/1673515526953-RoleColor.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								packages/backend/migration/1673515526953-RoleColor.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| export class RoleColor1673515526953 { | ||||
|     name = 'RoleColor1673515526953' | ||||
|  | ||||
|     async up(queryRunner) { | ||||
|         await queryRunner.query(`ALTER TABLE "role" ADD "color" character varying(256)`); | ||||
|     } | ||||
|  | ||||
|     async down(queryRunner) { | ||||
|         await queryRunner.query(`ALTER TABLE "role" DROP COLUMN "color"`); | ||||
|     } | ||||
| } | ||||
							
								
								
									
										13
									
								
								packages/backend/migration/1673522856499-RoleIroiro.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								packages/backend/migration/1673522856499-RoleIroiro.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| export class RoleIroiro1673522856499 { | ||||
|     name = 'RoleIroiro1673522856499' | ||||
|  | ||||
|     async up(queryRunner) { | ||||
|         await queryRunner.query(`ALTER TABLE "user" DROP COLUMN "isSilenced"`); | ||||
|         await queryRunner.query(`ALTER TABLE "role" ADD "canEditMembersByModerator" boolean NOT NULL DEFAULT false`); | ||||
|     } | ||||
|  | ||||
|     async down(queryRunner) { | ||||
|         await queryRunner.query(`ALTER TABLE "role" DROP COLUMN "canEditMembersByModerator"`); | ||||
|         await queryRunner.query(`ALTER TABLE "user" ADD "isSilenced" boolean NOT NULL DEFAULT false`); | ||||
|     } | ||||
| } | ||||
							
								
								
									
										13
									
								
								packages/backend/migration/1673524604156-RoleLastUsedAt.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								packages/backend/migration/1673524604156-RoleLastUsedAt.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| export class RoleLastUsedAt1673524604156 { | ||||
|     name = 'RoleLastUsedAt1673524604156' | ||||
|  | ||||
|     async up(queryRunner) { | ||||
|         await queryRunner.query(`ALTER TABLE "role" ADD "lastUsedAt" TIMESTAMP WITH TIME ZONE NOT NULL`); | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "role"."lastUsedAt" IS 'The last used date of the Role.'`); | ||||
|     } | ||||
|  | ||||
|     async down(queryRunner) { | ||||
|         await queryRunner.query(`COMMENT ON COLUMN "role"."lastUsedAt" IS 'The last used date of the Role.'`); | ||||
|         await queryRunner.query(`ALTER TABLE "role" DROP COLUMN "lastUsedAt"`); | ||||
|     } | ||||
| } | ||||
							
								
								
									
										15
									
								
								packages/backend/migration/1673570377815-RoleConditional.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								packages/backend/migration/1673570377815-RoleConditional.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | ||||
| export class RoleConditional1673570377815 { | ||||
|     name = 'RoleConditional1673570377815' | ||||
|  | ||||
|     async up(queryRunner) { | ||||
|         await queryRunner.query(`CREATE TYPE "public"."role_target_enum" AS ENUM('manual', 'conditional')`); | ||||
|         await queryRunner.query(`ALTER TABLE "role" ADD "target" "public"."role_target_enum" NOT NULL DEFAULT 'manual'`); | ||||
|         await queryRunner.query(`ALTER TABLE "role" ADD "condFormula" jsonb NOT NULL DEFAULT '{}'`); | ||||
|     } | ||||
|  | ||||
|     async down(queryRunner) { | ||||
|         await queryRunner.query(`ALTER TABLE "role" DROP COLUMN "condFormula"`); | ||||
|         await queryRunner.query(`ALTER TABLE "role" DROP COLUMN "target"`); | ||||
|         await queryRunner.query(`DROP TYPE "public"."role_target_enum"`); | ||||
|     } | ||||
| } | ||||
							
								
								
									
										11
									
								
								packages/backend/migration/1673575973645-MetaClean.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								packages/backend/migration/1673575973645-MetaClean.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| export class MetaClean1673575973645 { | ||||
|     name = 'MetaClean1673575973645' | ||||
|  | ||||
|     async up(queryRunner) { | ||||
|         await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "remoteDriveCapacityMb"`); | ||||
|     } | ||||
|  | ||||
|     async down(queryRunner) { | ||||
|         await queryRunner.query(`ALTER TABLE "meta" ADD "remoteDriveCapacityMb" integer NOT NULL DEFAULT '32'`); | ||||
|     } | ||||
| } | ||||
							
								
								
									
										13
									
								
								packages/backend/migration/1673783015567-Policies.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								packages/backend/migration/1673783015567-Policies.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| export class Policies1673783015567 { | ||||
|     name = 'Policies1673783015567' | ||||
|  | ||||
|     async up(queryRunner) { | ||||
|         await queryRunner.query(`ALTER TABLE "role" RENAME COLUMN "options" TO "policies"`); | ||||
| 				await queryRunner.query(`ALTER TABLE "meta" RENAME COLUMN "defaultRoleOverride" TO "policies"`); | ||||
|     } | ||||
|  | ||||
|     async down(queryRunner) { | ||||
| 				await queryRunner.query(`ALTER TABLE "meta" RENAME COLUMN "policies" TO "defaultRoleOverride"`); | ||||
|         await queryRunner.query(`ALTER TABLE "role" RENAME COLUMN "policies" TO "options"`); | ||||
|     } | ||||
| } | ||||
							
								
								
									
										11
									
								
								packages/backend/migration/1673812883772-firstRetrievedAt.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								packages/backend/migration/1673812883772-firstRetrievedAt.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| export class firstRetrievedAt1673812883772 { | ||||
|     name = 'firstRetrievedAt1673812883772' | ||||
|  | ||||
|     async up(queryRunner) { | ||||
|         await queryRunner.query(`ALTER TABLE "instance" RENAME COLUMN "caughtAt" TO "firstRetrievedAt"`); | ||||
|     } | ||||
|  | ||||
|     async down(queryRunner) { | ||||
|         await queryRunner.query(`ALTER TABLE "instance" RENAME COLUMN "firstRetrievedAt" TO "caughtAt"`); | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,11 @@ | ||||
| export class flashScriptLength1674086433654 { | ||||
|     name = 'flashScriptLength1674086433654' | ||||
|  | ||||
|     async up(queryRunner) { | ||||
|         await queryRunner.query(`ALTER TABLE "flash" ALTER COLUMN "script" TYPE character varying(32768)`); | ||||
|     } | ||||
|  | ||||
|     async down(queryRunner) { | ||||
| 			await queryRunner.query(`ALTER TABLE "flash" ALTER COLUMN "script" TYPE character varying(16384)`); | ||||
|     } | ||||
| } | ||||
							
								
								
									
										33
									
								
								packages/backend/migration/1674118260469-achievement.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								packages/backend/migration/1674118260469-achievement.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | ||||
| export class achievement1674118260469 { | ||||
|     name = 'achievement1674118260469' | ||||
|  | ||||
|     async up(queryRunner) { | ||||
|         await queryRunner.query(`ALTER TABLE "notification" ADD "achievement" character varying(128)`); | ||||
|         await queryRunner.query(`ALTER TABLE "user_profile" ADD "achievements" jsonb NOT NULL DEFAULT '[]'`); | ||||
|         await queryRunner.query(`ALTER TYPE "public"."notification_type_enum" RENAME TO "notification_type_enum_old"`); | ||||
|         await queryRunner.query(`CREATE TYPE "public"."notification_type_enum" AS ENUM('follow', 'mention', 'reply', 'renote', 'quote', 'reaction', 'pollVote', 'pollEnded', 'receiveFollowRequest', 'followRequestAccepted', 'groupInvited', 'achievementEarned', 'app')`); | ||||
|         await queryRunner.query(`ALTER TABLE "notification" ALTER COLUMN "type" TYPE "public"."notification_type_enum" USING "type"::"text"::"public"."notification_type_enum"`); | ||||
|         await queryRunner.query(`DROP TYPE "public"."notification_type_enum_old"`); | ||||
|         await queryRunner.query(`ALTER TYPE "public"."user_profile_mutingnotificationtypes_enum" RENAME TO "user_profile_mutingnotificationtypes_enum_old"`); | ||||
|         await queryRunner.query(`CREATE TYPE "public"."user_profile_mutingnotificationtypes_enum" AS ENUM('follow', 'mention', 'reply', 'renote', 'quote', 'reaction', 'pollVote', 'pollEnded', 'receiveFollowRequest', 'followRequestAccepted', 'groupInvited', 'achievementEarned', 'app')`); | ||||
|         await queryRunner.query(`ALTER TABLE "user_profile" ALTER COLUMN "mutingNotificationTypes" DROP DEFAULT`); | ||||
|         await queryRunner.query(`ALTER TABLE "user_profile" ALTER COLUMN "mutingNotificationTypes" TYPE "public"."user_profile_mutingnotificationtypes_enum"[] USING "mutingNotificationTypes"::"text"::"public"."user_profile_mutingnotificationtypes_enum"[]`); | ||||
|         await queryRunner.query(`ALTER TABLE "user_profile" ALTER COLUMN "mutingNotificationTypes" SET DEFAULT '{}'`); | ||||
|         await queryRunner.query(`DROP TYPE "public"."user_profile_mutingnotificationtypes_enum_old"`); | ||||
|     } | ||||
|  | ||||
|     async down(queryRunner) { | ||||
|         await queryRunner.query(`CREATE TYPE "public"."user_profile_mutingnotificationtypes_enum_old" AS ENUM('follow', 'mention', 'reply', 'renote', 'quote', 'reaction', 'pollVote', 'receiveFollowRequest', 'followRequestAccepted', 'groupInvited', 'app', 'pollEnded')`); | ||||
|         await queryRunner.query(`ALTER TABLE "user_profile" ALTER COLUMN "mutingNotificationTypes" DROP DEFAULT`); | ||||
|         await queryRunner.query(`ALTER TABLE "user_profile" ALTER COLUMN "mutingNotificationTypes" TYPE "public"."user_profile_mutingnotificationtypes_enum_old"[] USING "mutingNotificationTypes"::"text"::"public"."user_profile_mutingnotificationtypes_enum_old"[]`); | ||||
|         await queryRunner.query(`ALTER TABLE "user_profile" ALTER COLUMN "mutingNotificationTypes" SET DEFAULT '{}'`); | ||||
|         await queryRunner.query(`DROP TYPE "public"."user_profile_mutingnotificationtypes_enum"`); | ||||
|         await queryRunner.query(`ALTER TYPE "public"."user_profile_mutingnotificationtypes_enum_old" RENAME TO "user_profile_mutingnotificationtypes_enum"`); | ||||
|         await queryRunner.query(`CREATE TYPE "public"."notification_type_enum_old" AS ENUM('follow', 'mention', 'reply', 'renote', 'quote', 'reaction', 'pollVote', 'pollEnded', 'receiveFollowRequest', 'followRequestAccepted', 'groupInvited', 'app')`); | ||||
|         await queryRunner.query(`ALTER TABLE "notification" ALTER COLUMN "type" TYPE "public"."notification_type_enum_old" USING "type"::"text"::"public"."notification_type_enum_old"`); | ||||
|         await queryRunner.query(`DROP TYPE "public"."notification_type_enum"`); | ||||
|         await queryRunner.query(`ALTER TYPE "public"."notification_type_enum_old" RENAME TO "notification_type_enum"`); | ||||
|         await queryRunner.query(`ALTER TABLE "user_profile" DROP COLUMN "achievements"`); | ||||
|         await queryRunner.query(`ALTER TABLE "notification" DROP COLUMN "achievement"`); | ||||
|     } | ||||
| } | ||||
							
								
								
									
										11
									
								
								packages/backend/migration/1674255666603-loggedInDates.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								packages/backend/migration/1674255666603-loggedInDates.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| export class loggedInDates1674255666603 { | ||||
|     name = 'loggedInDates1674255666603' | ||||
|  | ||||
|     async up(queryRunner) { | ||||
|         await queryRunner.query(`ALTER TABLE "user_profile" ADD "loggedInDates" character varying(32) array NOT NULL DEFAULT '{}'`); | ||||
|     } | ||||
|  | ||||
|     async down(queryRunner) { | ||||
|         await queryRunner.query(`ALTER TABLE "user_profile" DROP COLUMN "loggedInDates"`); | ||||
|     } | ||||
| } | ||||
| @@ -6,32 +6,34 @@ | ||||
| 	"scripts": { | ||||
| 		"start": "node ./built/index.js", | ||||
| 		"start:test": "NODE_ENV=test node ./built/index.js", | ||||
| 		"migrate": "typeorm migration:run -d ormconfig.js", | ||||
| 		"migrate": "pnpm typeorm migration:run -d ormconfig.js", | ||||
| 		"build:swc": "swc src -d built -D", | ||||
| 		"watch:swc": "swc src -d built -D -w", | ||||
| 		"build": "tsc -p tsconfig.json || echo done. && tsc-alias -p tsconfig.json", | ||||
| 		"watch": "node watch.mjs", | ||||
| 		"lint": "tsc --noEmit && eslint --quiet \"src/**/*.ts\"", | ||||
| 		"jest": "cross-env NODE_ENV=test node --experimental-vm-modules --experimental-import-meta-resolve node_modules/jest/bin/jest.js --forceExit --runInBand", | ||||
| 		"jest-and-coverage": "cross-env NODE_ENV=test node --experimental-vm-modules --experimental-import-meta-resolve node_modules/jest/bin/jest.js --coverage --forceExit --runInBand", | ||||
| 		"jest-clear": "cross-env NODE_ENV=test node --experimental-vm-modules --experimental-import-meta-resolve node_modules/jest/bin/jest.js --clearCache", | ||||
| 		"test": "yarn jest", | ||||
| 		"test-and-coverage": "yarn jest-and-coverage" | ||||
| 		"test": "pnpm jest", | ||||
| 		"test-and-coverage": "pnpm jest-and-coverage" | ||||
| 	}, | ||||
| 	"optionalDependencies": { | ||||
| 		"@tensorflow/tfjs": "^4.1.0", | ||||
| 		"@tensorflow/tfjs-node": "4.1.0" | ||||
| 	}, | ||||
| 	"dependencies": { | ||||
| 		"@bull-board/api": "^4.10.1", | ||||
| 		"@bull-board/fastify": "^4.10.1", | ||||
| 		"@bull-board/ui": "^4.10.1", | ||||
| 		"@bull-board/api": "^4.10.2", | ||||
| 		"@bull-board/fastify": "^4.10.2", | ||||
| 		"@bull-board/ui": "^4.10.2", | ||||
| 		"@discordapp/twemoji": "14.0.2", | ||||
| 		"@fastify/accepts": "4.1.0", | ||||
| 		"@fastify/cookie": "^8.3.0", | ||||
| 		"@fastify/cors": "8.2.0", | ||||
| 		"@fastify/http-proxy": "^8.4.0", | ||||
| 		"@fastify/multipart": "7.3.0", | ||||
| 		"@fastify/static": "6.6.0", | ||||
| 		"@fastify/view": "7.3.0", | ||||
| 		"@fastify/multipart": "7.4.0", | ||||
| 		"@fastify/static": "6.6.1", | ||||
| 		"@fastify/view": "7.4.0", | ||||
| 		"@nestjs/common": "9.2.1", | ||||
| 		"@nestjs/core": "9.2.1", | ||||
| 		"@nestjs/testing": "9.2.1", | ||||
| @@ -41,7 +43,7 @@ | ||||
| 		"ajv": "8.12.0", | ||||
| 		"archiver": "5.3.1", | ||||
| 		"autwh": "0.1.0", | ||||
| 		"aws-sdk": "2.1289.0", | ||||
| 		"aws-sdk": "2.1295.0", | ||||
| 		"bcryptjs": "2.4.3", | ||||
| 		"blurhash": "2.0.4", | ||||
| 		"bull": "4.10.2", | ||||
| @@ -58,7 +60,7 @@ | ||||
| 		"escape-regexp": "0.0.1", | ||||
| 		"fastify": "4.11.0", | ||||
| 		"feed": "4.2.2", | ||||
| 		"file-type": "18.0.0", | ||||
| 		"file-type": "18.1.0", | ||||
| 		"fluent-ffmpeg": "2.1.2", | ||||
| 		"form-data": "^4.0.0", | ||||
| 		"got": "12.5.3", | ||||
| @@ -67,18 +69,17 @@ | ||||
| 		"ip-cidr": "3.0.11", | ||||
| 		"is-svg": "4.3.2", | ||||
| 		"js-yaml": "4.1.0", | ||||
| 		"jsdom": "20.0.3", | ||||
| 		"jsdom": "21.0.0", | ||||
| 		"json5": "2.2.3", | ||||
| 		"json5-loader": "4.0.1", | ||||
| 		"jsonld": "8.1.0", | ||||
| 		"jsrsasign": "10.6.1", | ||||
| 		"mfm-js": "0.23.1", | ||||
| 		"mfm-js": "0.23.3", | ||||
| 		"mime-types": "2.1.35", | ||||
| 		"misskey-js": "0.0.14", | ||||
| 		"ms": "3.0.0-canary.1", | ||||
| 		"nested-property": "4.0.0", | ||||
| 		"node-fetch": "3.3.0", | ||||
| 		"nodemailer": "6.8.0", | ||||
| 		"nodemailer": "6.9.0", | ||||
| 		"nsfwjs": "2.4.2", | ||||
| 		"oauth": "^0.10.0", | ||||
| 		"os-utils": "0.0.14", | ||||
| @@ -88,7 +89,7 @@ | ||||
| 		"probe-image-size": "7.2.3", | ||||
| 		"promise-limit": "2.7.0", | ||||
| 		"pug": "3.0.2", | ||||
| 		"punycode": "2.1.1", | ||||
| 		"punycode": "2.2.0", | ||||
| 		"pureimage": "0.3.15", | ||||
| 		"qrcode": "1.5.1", | ||||
| 		"random-seed": "0.3.0", | ||||
| @@ -110,25 +111,28 @@ | ||||
| 		"stringz": "2.1.0", | ||||
| 		"summaly": "2.7.0", | ||||
| 		"syslog-pro": "git+https://github.com/misskey-dev/SyslogPro#0.2.9-misskey.2", | ||||
| 		"systeminformation": "5.17.1", | ||||
| 		"systeminformation": "5.17.3", | ||||
| 		"tinycolor2": "1.5.2", | ||||
| 		"tmp": "0.2.1", | ||||
| 		"tsc-alias": "1.8.2", | ||||
| 		"tsconfig-paths": "4.1.2", | ||||
| 		"twemoji-parser": "14.0.0", | ||||
| 		"typeorm": "0.3.11", | ||||
| 		"typescript": "4.9.4", | ||||
| 		"ulid": "2.3.0", | ||||
| 		"undici": "^5.15.0", | ||||
| 		"unzipper": "0.10.11", | ||||
| 		"uuid": "9.0.0", | ||||
| 		"vary": "1.1.2", | ||||
| 		"web-push": "3.5.0", | ||||
| 		"websocket": "1.0.34", | ||||
| 		"ws": "8.11.0", | ||||
| 		"ws": "8.12.0", | ||||
| 		"xev": "3.0.2" | ||||
| 	}, | ||||
| 	"devDependencies": { | ||||
| 		"@redocly/openapi-core": "1.0.0-beta.117", | ||||
| 		"@swc/core": "1.3.25", | ||||
| 		"@redocly/openapi-core": "1.0.0-beta.120", | ||||
| 		"@swc/cli": "^0.1.59", | ||||
| 		"@swc/core": "1.3.26", | ||||
| 		"@swc/jest": "0.2.24", | ||||
| 		"@types/accepts": "1.3.5", | ||||
| 		"@types/archiver": "5.3.1", | ||||
| @@ -172,14 +176,14 @@ | ||||
| 		"@types/web-push": "3.3.2", | ||||
| 		"@types/websocket": "1.0.5", | ||||
| 		"@types/ws": "8.5.4", | ||||
| 		"@typescript-eslint/eslint-plugin": "5.48.0", | ||||
| 		"@typescript-eslint/parser": "5.48.0", | ||||
| 		"@typescript-eslint/eslint-plugin": "5.48.1", | ||||
| 		"@typescript-eslint/parser": "5.48.1", | ||||
| 		"cross-env": "7.0.3", | ||||
| 		"eslint": "8.31.0", | ||||
| 		"eslint-plugin-import": "2.26.0", | ||||
| 		"eslint-plugin-import": "2.27.4", | ||||
| 		"execa": "6.1.0", | ||||
| 		"jest": "29.3.1", | ||||
| 		"jest-mock": "^29.3.1", | ||||
| 		"typescript": "4.9.4" | ||||
| 		"node-fetch": "3.3.0" | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -1,13 +1,13 @@ | ||||
| import { Module } from '@nestjs/common'; | ||||
| import { ServerModule } from '@/server/ServerModule.js'; | ||||
| import { GlobalModule } from '@/GlobalModule.js'; | ||||
| import { QueueProcessorModule } from '@/queue/QueueProcessorModule.js'; | ||||
| import { DaemonModule } from '@/daemons/DaemonModule.js'; | ||||
| 
 | ||||
| @Module({ | ||||
| 	imports: [ | ||||
| 		GlobalModule, | ||||
| 		ServerModule, | ||||
| 		QueueProcessorModule, | ||||
| 		DaemonModule, | ||||
| 	], | ||||
| }) | ||||
| export class RootModule {} | ||||
| export class MainModule {} | ||||
| @@ -17,6 +17,9 @@ import { JanitorService } from '@/daemons/JanitorService.js'; | ||||
| import { QueueStatsService } from '@/daemons/QueueStatsService.js'; | ||||
| import { ServerStatsService } from '@/daemons/ServerStatsService.js'; | ||||
| import { NestLogger } from '@/NestLogger.js'; | ||||
| import { ChartManagementService } from '@/core/chart/ChartManagementService.js'; | ||||
| import { ServerService } from '@/server/ServerService.js'; | ||||
| import { MainModule } from '@/MainModule.js'; | ||||
| import { envOption } from '../env.js'; | ||||
|  | ||||
| const _filename = fileURLToPath(import.meta.url); | ||||
| @@ -70,6 +73,15 @@ export async function masterMain() { | ||||
| 		process.exit(1); | ||||
| 	} | ||||
|  | ||||
| 	const app = await NestFactory.createApplicationContext(MainModule, { | ||||
| 		logger: new NestLogger(), | ||||
| 	}); | ||||
| 	app.enableShutdownHooks(); | ||||
|  | ||||
| 	// start server | ||||
| 	const serverService = app.get(ServerService); | ||||
| 	serverService.launch(); | ||||
|  | ||||
| 	bootLogger.succ('Misskey initialized'); | ||||
|  | ||||
| 	if (!envOption.disableClustering) { | ||||
| @@ -78,15 +90,10 @@ export async function masterMain() { | ||||
|  | ||||
| 	bootLogger.succ(`Now listening on port ${config.port} on ${config.url}`, null, true); | ||||
|  | ||||
| 	if (!envOption.noDaemons) { | ||||
| 		const daemons = await NestFactory.createApplicationContext(DaemonModule, { | ||||
| 			logger: new NestLogger(), | ||||
| 		}); | ||||
| 		daemons.enableShutdownHooks(); | ||||
| 		daemons.get(JanitorService).start(); | ||||
| 		daemons.get(QueueStatsService).start(); | ||||
| 		daemons.get(ServerStatsService).start(); | ||||
| 	} | ||||
| 	app.get(ChartManagementService).start(); | ||||
| 	app.get(JanitorService).start(); | ||||
| 	app.get(QueueStatsService).start(); | ||||
| 	app.get(ServerStatsService).start(); | ||||
| } | ||||
|  | ||||
| function showEnvironment(): void { | ||||
|   | ||||
| @@ -1,32 +1,23 @@ | ||||
| import cluster from 'node:cluster'; | ||||
| import { NestFactory } from '@nestjs/core'; | ||||
| import { envOption } from '@/env.js'; | ||||
| import { ChartManagementService } from '@/core/chart/ChartManagementService.js'; | ||||
| import { ServerService } from '@/server/ServerService.js'; | ||||
| import { QueueProcessorService } from '@/queue/QueueProcessorService.js'; | ||||
| import { NestLogger } from '@/NestLogger.js'; | ||||
| import { RootModule } from '../RootModule.js'; | ||||
| import { QueueProcessorModule } from '@/queue/QueueProcessorModule.js'; | ||||
|  | ||||
| /** | ||||
|  * Init worker process | ||||
|  */ | ||||
| export async function workerMain() { | ||||
| 	const app = await NestFactory.createApplicationContext(RootModule, { | ||||
| 	const jobQueue = await NestFactory.createApplicationContext(QueueProcessorModule, { | ||||
| 		logger: new NestLogger(), | ||||
| 	}); | ||||
| 	app.enableShutdownHooks(); | ||||
|  | ||||
| 	// start server | ||||
| 	const serverService = app.get(ServerService); | ||||
| 	serverService.launch(); | ||||
| 	jobQueue.enableShutdownHooks(); | ||||
|  | ||||
| 	// start job queue | ||||
| 	if (!envOption.onlyServer) { | ||||
| 		const queueProcessorService = app.get(QueueProcessorService); | ||||
| 		queueProcessorService.start(); | ||||
| 	} | ||||
| 	jobQueue.get(QueueProcessorService).start(); | ||||
|  | ||||
| 	app.get(ChartManagementService).run(); | ||||
| 	jobQueue.get(ChartManagementService).start(); | ||||
|  | ||||
| 	if (cluster.isWorker) { | ||||
| 		// Send a 'ready' message to parent process | ||||
|   | ||||
							
								
								
									
										114
									
								
								packages/backend/src/core/AchievementService.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										114
									
								
								packages/backend/src/core/AchievementService.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,114 @@ | ||||
| import { Inject, Injectable } from '@nestjs/common'; | ||||
| import type { UserProfilesRepository, UsersRepository } from '@/models/index.js'; | ||||
| import type { User } from '@/models/entities/User.js'; | ||||
| import { DI } from '@/di-symbols.js'; | ||||
| import { bindThis } from '@/decorators.js'; | ||||
| import { CreateNotificationService } from '@/core/CreateNotificationService.js'; | ||||
|  | ||||
| const ACHIEVEMENT_TYPES = [ | ||||
| 	'notes1', | ||||
| 	'notes10', | ||||
| 	'notes100', | ||||
| 	'notes500', | ||||
| 	'notes1000', | ||||
| 	'notes5000', | ||||
| 	'notes10000', | ||||
| 	'notes20000', | ||||
| 	'notes30000', | ||||
| 	'notes40000', | ||||
| 	'notes50000', | ||||
| 	'notes60000', | ||||
| 	'notes70000', | ||||
| 	'notes80000', | ||||
| 	'notes90000', | ||||
| 	'notes100000', | ||||
| 	'login3', | ||||
| 	'login7', | ||||
| 	'login15', | ||||
| 	'login30', | ||||
| 	'login60', | ||||
| 	'login100', | ||||
| 	'login200', | ||||
| 	'login300', | ||||
| 	'login400', | ||||
| 	'login500', | ||||
| 	'login600', | ||||
| 	'login700', | ||||
| 	'login800', | ||||
| 	'login900', | ||||
| 	'login1000', | ||||
| 	'passedSinceAccountCreated1', | ||||
| 	'passedSinceAccountCreated2', | ||||
| 	'passedSinceAccountCreated3', | ||||
| 	'loggedInOnBirthday', | ||||
| 	'noteClipped1', | ||||
| 	'noteFavorited1', | ||||
| 	'profileFilled', | ||||
| 	'markedAsCat', | ||||
| 	'following1', | ||||
| 	'following10', | ||||
| 	'following50', | ||||
| 	'following100', | ||||
| 	'following300', | ||||
| 	'followers1', | ||||
| 	'followers10', | ||||
| 	'followers50', | ||||
| 	'followers100', | ||||
| 	'followers300', | ||||
| 	'followers500', | ||||
| 	'followers1000', | ||||
| 	'collectAchievements30', | ||||
| 	'iLoveMisskey', | ||||
| 	'client30min', | ||||
| 	'noteDeletedWithin1min', | ||||
| 	'postedAtLateNight', | ||||
| 	'postedAt0min0sec', | ||||
| 	'selfQuote', | ||||
| 	'htl20npm', | ||||
| 	'driveFolderCircularReference', | ||||
| 	'reactWithoutRead', | ||||
| 	'clickedClickHere', | ||||
| 	'justPlainLucky', | ||||
| 	'setNameToSyuilo', | ||||
| 	'cookieClicked', | ||||
| 	'brainDiver', | ||||
| ] as const; | ||||
|  | ||||
| @Injectable() | ||||
| export class AchievementService { | ||||
| 	constructor( | ||||
| 		@Inject(DI.usersRepository) | ||||
| 		private usersRepository: UsersRepository, | ||||
|  | ||||
| 		@Inject(DI.userProfilesRepository) | ||||
| 		private userProfilesRepository: UserProfilesRepository, | ||||
|  | ||||
| 		private createNotificationService: CreateNotificationService, | ||||
| 	) { | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public async create( | ||||
| 		userId: User['id'], | ||||
| 		type: string, | ||||
| 	): Promise<void> { | ||||
| 		if (!ACHIEVEMENT_TYPES.includes(type)) return; | ||||
|  | ||||
| 		const date = Date.now(); | ||||
|  | ||||
| 		const profile = await this.userProfilesRepository.findOneByOrFail({ userId: userId }); | ||||
|  | ||||
| 		if (profile.achievements.some(a => a.name === type)) return; | ||||
|  | ||||
| 		await this.userProfilesRepository.update(userId, { | ||||
| 			achievements: [...profile.achievements, { | ||||
| 				name: type, | ||||
| 				unlockedAt: date, | ||||
| 			}], | ||||
| 		}); | ||||
|  | ||||
| 		this.createNotificationService.createNotification(userId, 'achievementEarned', { | ||||
| 			achievement: type, | ||||
| 		}); | ||||
| 	} | ||||
| } | ||||
| @@ -16,6 +16,7 @@ import { DI } from '@/di-symbols.js'; | ||||
| import type { MutingsRepository, BlockingsRepository, NotesRepository, AntennaNotesRepository, AntennasRepository, UserGroupJoiningsRepository, UserListJoiningsRepository } from '@/models/index.js'; | ||||
| import { UtilityService } from '@/core/UtilityService.js'; | ||||
| import { bindThis } from '@/decorators.js'; | ||||
| import { StreamMessages } from '@/server/api/stream/types.js'; | ||||
| import type { OnApplicationShutdown } from '@nestjs/common'; | ||||
|  | ||||
| @Injectable() | ||||
| @@ -73,7 +74,7 @@ export class AntennaService implements OnApplicationShutdown { | ||||
| 		const obj = JSON.parse(data); | ||||
|  | ||||
| 		if (obj.channel === 'internal') { | ||||
| 			const { type, body } = obj.message; | ||||
| 			const { type, body } = obj.message as StreamMessages['internal']['payload']; | ||||
| 			switch (type) { | ||||
| 				case 'antennaCreated': | ||||
| 					this.antennas.push(body); | ||||
|   | ||||
| @@ -1,7 +1,4 @@ | ||||
| import { Inject, Injectable } from '@nestjs/common'; | ||||
| import { DI } from '@/di-symbols.js'; | ||||
| import type { UsersRepository } from '@/models/index.js'; | ||||
| import type { Config } from '@/config.js'; | ||||
| import { Injectable } from '@nestjs/common'; | ||||
| import { HttpRequestService } from '@/core/HttpRequestService.js'; | ||||
| import { bindThis } from '@/decorators.js'; | ||||
|  | ||||
| @@ -13,9 +10,6 @@ type CaptchaResponse = { | ||||
| @Injectable() | ||||
| export class CaptchaService { | ||||
| 	constructor( | ||||
| 		@Inject(DI.config) | ||||
| 		private config: Config, | ||||
|  | ||||
| 		private httpRequestService: HttpRequestService, | ||||
| 	) { | ||||
| 	} | ||||
| @@ -27,16 +21,16 @@ export class CaptchaService { | ||||
| 			response, | ||||
| 		}); | ||||
| 	 | ||||
| 		const res = await fetch(url, { | ||||
| 			method: 'POST', | ||||
| 			body: params, | ||||
| 			headers: { | ||||
| 				'User-Agent': this.config.userAgent, | ||||
| 		const res = await this.httpRequestService.fetch( | ||||
| 			url, | ||||
| 			{ | ||||
| 				method: 'POST', | ||||
| 				body: params, | ||||
| 			}, | ||||
| 			// TODO | ||||
| 			//timeout: 10 * 1000, | ||||
| 			agent: (url, bypassProxy) => this.httpRequestService.getAgentByUrl(url, bypassProxy), | ||||
| 		}).catch(err => { | ||||
| 			{ | ||||
| 				noOkError: true, | ||||
| 			} | ||||
| 		).catch(err => { | ||||
| 			throw `${err.message ?? err}`; | ||||
| 		}); | ||||
| 	 | ||||
|   | ||||
| @@ -4,6 +4,7 @@ import { AccountUpdateService } from './AccountUpdateService.js'; | ||||
| import { AiService } from './AiService.js'; | ||||
| import { AntennaService } from './AntennaService.js'; | ||||
| import { AppLockService } from './AppLockService.js'; | ||||
| import { AchievementService } from './AchievementService.js'; | ||||
| import { CaptchaService } from './CaptchaService.js'; | ||||
| import { CreateNotificationService } from './CreateNotificationService.js'; | ||||
| import { CreateSystemUserService } from './CreateSystemUserService.js'; | ||||
| @@ -35,6 +36,7 @@ import { PushNotificationService } from './PushNotificationService.js'; | ||||
| import { QueryService } from './QueryService.js'; | ||||
| import { ReactionService } from './ReactionService.js'; | ||||
| import { RelayService } from './RelayService.js'; | ||||
| import { RoleService } from './RoleService.js'; | ||||
| import { S3Service } from './S3Service.js'; | ||||
| import { SignupService } from './SignupService.js'; | ||||
| import { TwoFactorAuthenticationService } from './TwoFactorAuthenticationService.js'; | ||||
| @@ -97,6 +99,7 @@ import { UserGroupInvitationEntityService } from './entities/UserGroupInvitation | ||||
| import { UserListEntityService } from './entities/UserListEntityService.js'; | ||||
| import { FlashEntityService } from './entities/FlashEntityService.js'; | ||||
| import { FlashLikeEntityService } from './entities/FlashLikeEntityService.js'; | ||||
| import { RoleEntityService } from './entities/RoleEntityService.js'; | ||||
| import { ApAudienceService } from './activitypub/ApAudienceService.js'; | ||||
| import { ApDbResolverService } from './activitypub/ApDbResolverService.js'; | ||||
| import { ApDeliverManagerService } from './activitypub/ApDeliverManagerService.js'; | ||||
| @@ -126,6 +129,7 @@ const $AccountUpdateService: Provider = { provide: 'AccountUpdateService', useEx | ||||
| const $AiService: Provider = { provide: 'AiService', useExisting: AiService }; | ||||
| const $AntennaService: Provider = { provide: 'AntennaService', useExisting: AntennaService }; | ||||
| const $AppLockService: Provider = { provide: 'AppLockService', useExisting: AppLockService }; | ||||
| const $AchievementService: Provider = { provide: 'AchievementService', useExisting: AchievementService }; | ||||
| const $CaptchaService: Provider = { provide: 'CaptchaService', useExisting: CaptchaService }; | ||||
| const $CreateNotificationService: Provider = { provide: 'CreateNotificationService', useExisting: CreateNotificationService }; | ||||
| const $CreateSystemUserService: Provider = { provide: 'CreateSystemUserService', useExisting: CreateSystemUserService }; | ||||
| @@ -158,6 +162,7 @@ const $PushNotificationService: Provider = { provide: 'PushNotificationService', | ||||
| const $QueryService: Provider = { provide: 'QueryService', useExisting: QueryService }; | ||||
| const $ReactionService: Provider = { provide: 'ReactionService', useExisting: ReactionService }; | ||||
| const $RelayService: Provider = { provide: 'RelayService', useExisting: RelayService }; | ||||
| const $RoleService: Provider = { provide: 'RoleService', useExisting: RoleService }; | ||||
| const $S3Service: Provider = { provide: 'S3Service', useExisting: S3Service }; | ||||
| const $SignupService: Provider = { provide: 'SignupService', useExisting: SignupService }; | ||||
| const $TwoFactorAuthenticationService: Provider = { provide: 'TwoFactorAuthenticationService', useExisting: TwoFactorAuthenticationService }; | ||||
| @@ -220,6 +225,7 @@ const $UserGroupInvitationEntityService: Provider = { provide: 'UserGroupInvitat | ||||
| const $UserListEntityService: Provider = { provide: 'UserListEntityService', useExisting: UserListEntityService }; | ||||
| const $FlashEntityService: Provider = { provide: 'FlashEntityService', useExisting: FlashEntityService }; | ||||
| const $FlashLikeEntityService: Provider = { provide: 'FlashLikeEntityService', useExisting: FlashLikeEntityService }; | ||||
| const $RoleEntityService: Provider = { provide: 'RoleEntityService', useExisting: RoleEntityService }; | ||||
|  | ||||
| const $ApAudienceService: Provider = { provide: 'ApAudienceService', useExisting: ApAudienceService }; | ||||
| const $ApDbResolverService: Provider = { provide: 'ApDbResolverService', useExisting: ApDbResolverService }; | ||||
| @@ -251,6 +257,7 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | ||||
| 		AiService, | ||||
| 		AntennaService, | ||||
| 		AppLockService, | ||||
| 		AchievementService, | ||||
| 		CaptchaService, | ||||
| 		CreateNotificationService, | ||||
| 		CreateSystemUserService, | ||||
| @@ -283,6 +290,7 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | ||||
| 		QueryService, | ||||
| 		ReactionService, | ||||
| 		RelayService, | ||||
| 		RoleService, | ||||
| 		S3Service, | ||||
| 		SignupService, | ||||
| 		TwoFactorAuthenticationService, | ||||
| @@ -344,6 +352,7 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | ||||
| 		UserListEntityService, | ||||
| 		FlashEntityService, | ||||
| 		FlashLikeEntityService, | ||||
| 		RoleEntityService, | ||||
| 		ApAudienceService, | ||||
| 		ApDbResolverService, | ||||
| 		ApDeliverManagerService, | ||||
| @@ -370,6 +379,7 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | ||||
| 		$AiService, | ||||
| 		$AntennaService, | ||||
| 		$AppLockService, | ||||
| 		$AchievementService, | ||||
| 		$CaptchaService, | ||||
| 		$CreateNotificationService, | ||||
| 		$CreateSystemUserService, | ||||
| @@ -402,6 +412,7 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | ||||
| 		$QueryService, | ||||
| 		$ReactionService, | ||||
| 		$RelayService, | ||||
| 		$RoleService, | ||||
| 		$S3Service, | ||||
| 		$SignupService, | ||||
| 		$TwoFactorAuthenticationService, | ||||
| @@ -463,6 +474,7 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | ||||
| 		$UserListEntityService, | ||||
| 		$FlashEntityService, | ||||
| 		$FlashLikeEntityService, | ||||
| 		$RoleEntityService, | ||||
| 		$ApAudienceService, | ||||
| 		$ApDbResolverService, | ||||
| 		$ApDeliverManagerService, | ||||
| @@ -490,6 +502,7 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | ||||
| 		AiService, | ||||
| 		AntennaService, | ||||
| 		AppLockService, | ||||
| 		AchievementService, | ||||
| 		CaptchaService, | ||||
| 		CreateNotificationService, | ||||
| 		CreateSystemUserService, | ||||
| @@ -522,6 +535,7 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | ||||
| 		QueryService, | ||||
| 		ReactionService, | ||||
| 		RelayService, | ||||
| 		RoleService, | ||||
| 		S3Service, | ||||
| 		SignupService, | ||||
| 		TwoFactorAuthenticationService, | ||||
| @@ -582,6 +596,7 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | ||||
| 		UserListEntityService, | ||||
| 		FlashEntityService, | ||||
| 		FlashLikeEntityService, | ||||
| 		RoleEntityService, | ||||
| 		ApAudienceService, | ||||
| 		ApDbResolverService, | ||||
| 		ApDeliverManagerService, | ||||
| @@ -608,6 +623,7 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | ||||
| 		$AiService, | ||||
| 		$AntennaService, | ||||
| 		$AppLockService, | ||||
| 		$AchievementService, | ||||
| 		$CaptchaService, | ||||
| 		$CreateNotificationService, | ||||
| 		$CreateSystemUserService, | ||||
| @@ -640,6 +656,7 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | ||||
| 		$QueryService, | ||||
| 		$ReactionService, | ||||
| 		$RelayService, | ||||
| 		$RoleService, | ||||
| 		$S3Service, | ||||
| 		$SignupService, | ||||
| 		$TwoFactorAuthenticationService, | ||||
| @@ -700,6 +717,7 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting | ||||
| 		$UserListEntityService, | ||||
| 		$FlashEntityService, | ||||
| 		$FlashLikeEntityService, | ||||
| 		$RoleEntityService, | ||||
| 		$ApAudienceService, | ||||
| 		$ApDbResolverService, | ||||
| 		$ApDeliverManagerService, | ||||
|   | ||||
| @@ -53,7 +53,7 @@ export class CreateSystemUserService { | ||||
| 				usernameLower: username.toLowerCase(), | ||||
| 				host: null, | ||||
| 				token: secret, | ||||
| 				isAdmin: false, | ||||
| 				isRoot: false, | ||||
| 				isLocked: true, | ||||
| 				isExplorable: false, | ||||
| 				isBot: true, | ||||
|   | ||||
| @@ -23,6 +23,9 @@ export class DeleteAccountService { | ||||
| 		id: string; | ||||
| 		host: string | null; | ||||
| 	}): Promise<void> { | ||||
| 		const _user = await this.usersRepository.findOneByOrFail({ id: user.id }); | ||||
| 		if (_user.isRoot) throw new Error('cannot delete a root account'); | ||||
|  | ||||
| 		// 物理削除する前にDelete activityを送信する | ||||
| 		await this.userSuspendService.doPostSuspend(user).catch(e => {}); | ||||
| 	 | ||||
|   | ||||
| @@ -8,11 +8,12 @@ import got, * as Got from 'got'; | ||||
| import chalk from 'chalk'; | ||||
| import { DI } from '@/di-symbols.js'; | ||||
| import type { Config } from '@/config.js'; | ||||
| import { HttpRequestService } from '@/core/HttpRequestService.js'; | ||||
| import { HttpRequestService, UndiciFetcher } from '@/core/HttpRequestService.js'; | ||||
| import { createTemp } from '@/misc/create-temp.js'; | ||||
| import { StatusError } from '@/misc/status-error.js'; | ||||
| import { LoggerService } from '@/core/LoggerService.js'; | ||||
| import type Logger from '@/logger.js'; | ||||
| import { buildConnector } from 'undici'; | ||||
|  | ||||
| const pipeline = util.promisify(stream.pipeline); | ||||
| import { bindThis } from '@/decorators.js'; | ||||
| @@ -20,6 +21,7 @@ import { bindThis } from '@/decorators.js'; | ||||
| @Injectable() | ||||
| export class DownloadService { | ||||
| 	private logger: Logger; | ||||
| 	private undiciFetcher: UndiciFetcher; | ||||
|  | ||||
| 	constructor( | ||||
| 		@Inject(DI.config) | ||||
| @@ -29,70 +31,42 @@ export class DownloadService { | ||||
| 		private loggerService: LoggerService, | ||||
| 	) { | ||||
| 		this.logger = this.loggerService.getLogger('download'); | ||||
|  | ||||
| 		this.undiciFetcher = new UndiciFetcher(this.httpRequestService.getStandardUndiciFetcherOption( | ||||
| 			{ | ||||
| 				connect: process.env.NODE_ENV === 'development' ? | ||||
| 					this.httpRequestService.clientDefaults.connect | ||||
| 					: | ||||
| 					this.httpRequestService.getConnectorWithIpCheck( | ||||
| 						buildConnector({ | ||||
| 							...this.httpRequestService.clientDefaults.connect, | ||||
| 						}), | ||||
| 						(ip) => !this.isPrivateIp(ip) | ||||
| 					), | ||||
| 				bodyTimeout: 30 * 1000, | ||||
| 			}, | ||||
| 			{ | ||||
| 				connect: this.httpRequestService.clientDefaults.connect, | ||||
| 			} | ||||
| 		), this.logger); | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public async downloadUrl(url: string, path: string): Promise<void> { | ||||
| 		this.logger.info(`Downloading ${chalk.cyan(url)} to ${chalk.cyanBright(path)} ...`); | ||||
| 	 | ||||
|  | ||||
| 		const timeout = 30 * 1000; | ||||
| 		const operationTimeout = 60 * 1000; | ||||
| 		const maxSize = this.config.maxFileSize ?? 262144000; | ||||
| 	 | ||||
| 		const req = got.stream(url, { | ||||
| 			headers: { | ||||
| 				'User-Agent': this.config.userAgent, | ||||
| 			}, | ||||
| 			timeout: { | ||||
| 				lookup: timeout, | ||||
| 				connect: timeout, | ||||
| 				secureConnect: timeout, | ||||
| 				socket: timeout,	// read timeout | ||||
| 				response: timeout, | ||||
| 				send: timeout, | ||||
| 				request: operationTimeout,	// whole operation timeout | ||||
| 			}, | ||||
| 			agent: { | ||||
| 				http: this.httpRequestService.httpAgent, | ||||
| 				https: this.httpRequestService.httpsAgent, | ||||
| 			}, | ||||
| 			http2: false,	// default | ||||
| 			retry: { | ||||
| 				limit: 0, | ||||
| 			}, | ||||
| 		}).on('response', (res: Got.Response) => { | ||||
| 			if ((process.env.NODE_ENV === 'production' || process.env.NODE_ENV === 'test') && !this.config.proxy && res.ip) { | ||||
| 				if (this.isPrivateIp(res.ip)) { | ||||
| 					this.logger.warn(`Blocked address: ${res.ip}`); | ||||
| 					req.destroy(); | ||||
| 				} | ||||
| 			} | ||||
| 	 | ||||
| 			const contentLength = res.headers['content-length']; | ||||
| 			if (contentLength != null) { | ||||
| 				const size = Number(contentLength); | ||||
| 				if (size > maxSize) { | ||||
| 					this.logger.warn(`maxSize exceeded (${size} > ${maxSize}) on response`); | ||||
| 					req.destroy(); | ||||
| 				} | ||||
| 			} | ||||
| 		}).on('downloadProgress', (progress: Got.Progress) => { | ||||
| 			if (progress.transferred > maxSize) { | ||||
| 				this.logger.warn(`maxSize exceeded (${progress.transferred} > ${maxSize}) on downloadProgress`); | ||||
| 				req.destroy(); | ||||
| 			} | ||||
| 		}); | ||||
| 	 | ||||
| 		try { | ||||
| 			await pipeline(req, fs.createWriteStream(path)); | ||||
| 		} catch (e) { | ||||
| 			if (e instanceof Got.HTTPError) { | ||||
| 				throw new StatusError(`${e.response.statusCode} ${e.response.statusMessage}`, e.response.statusCode, e.response.statusMessage); | ||||
| 			} else { | ||||
| 				throw e; | ||||
| 			} | ||||
|  | ||||
| 		const response = await this.undiciFetcher.fetch(url); | ||||
|  | ||||
| 		if (response.body === null) { | ||||
| 			throw new StatusError('No body', 400, 'No body'); | ||||
| 		} | ||||
| 	 | ||||
|  | ||||
| 		await pipeline(stream.Readable.fromWeb(response.body), fs.createWriteStream(path)); | ||||
|  | ||||
| 		this.logger.succ(`Download finished: ${chalk.cyan(url)}`); | ||||
| 	} | ||||
|  | ||||
| @@ -114,7 +88,7 @@ export class DownloadService { | ||||
| 			cleanup(); | ||||
| 		} | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	@bindThis | ||||
| 	private isPrivateIp(ip: string): boolean { | ||||
| 		for (const net of this.config.allowedPrivateNetworks ?? []) { | ||||
| @@ -124,6 +98,6 @@ export class DownloadService { | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		return PrivateIp(ip); | ||||
| 		return PrivateIp(ip) ?? false; | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -32,11 +32,12 @@ import { DriveFileEntityService } from '@/core/entities/DriveFileEntityService.j | ||||
| import { UserEntityService } from '@/core/entities/UserEntityService.js'; | ||||
| import { FileInfoService } from '@/core/FileInfoService.js'; | ||||
| import { bindThis } from '@/decorators.js'; | ||||
| import { RoleService } from '@/core/RoleService.js'; | ||||
| import type S3 from 'aws-sdk/clients/s3.js'; | ||||
|  | ||||
| type AddFileArgs = { | ||||
| 	/** User who wish to add file */ | ||||
| 	user: { id: User['id']; host: User['host']; driveCapacityOverrideMb: User['driveCapacityOverrideMb'] } | null; | ||||
| 	user: { id: User['id']; host: User['host'] } | null; | ||||
| 	/** File path */ | ||||
| 	path: string; | ||||
| 	/** Name */ | ||||
| @@ -62,7 +63,7 @@ type AddFileArgs = { | ||||
|  | ||||
| type UploadFromUrlArgs = { | ||||
| 	url: string; | ||||
| 	user: { id: User['id']; host: User['host']; driveCapacityOverrideMb: User['driveCapacityOverrideMb'] } | null; | ||||
| 	user: { id: User['id']; host: User['host'] } | null; | ||||
| 	folderId?: DriveFolder['id'] | null; | ||||
| 	uri?: string | null; | ||||
| 	sensitive?: boolean; | ||||
| @@ -106,6 +107,7 @@ export class DriveService { | ||||
| 		private videoProcessingService: VideoProcessingService, | ||||
| 		private globalEventService: GlobalEventService, | ||||
| 		private queueService: QueueService, | ||||
| 		private roleService: RoleService, | ||||
| 		private driveChart: DriveChart, | ||||
| 		private perUserDriveChart: PerUserDriveChart, | ||||
| 		private instanceChart: InstanceChart, | ||||
| @@ -373,8 +375,19 @@ export class DriveService { | ||||
| 			partSize: s3.endpoint.hostname === 'storage.googleapis.com' ? 500 * 1024 * 1024 : 8 * 1024 * 1024, | ||||
| 		}); | ||||
|  | ||||
| 		const result = await upload.promise(); | ||||
| 		if (result) this.registerLogger.debug(`Uploaded: ${result.Bucket}/${result.Key} => ${result.Location}`); | ||||
| 		await upload.promise() | ||||
| 			.then( | ||||
| 				result => { | ||||
| 					if (result) { | ||||
| 						this.registerLogger.debug(`Uploaded: ${result.Bucket}/${result.Key} => ${result.Location}`); | ||||
| 					} else { | ||||
| 						this.registerLogger.error(`Upload Result Empty: key = ${key}, filename = ${filename}`); | ||||
| 					} | ||||
| 				}, | ||||
| 				err => { | ||||
| 					this.registerLogger.error(`Upload Failed: key = ${key}, filename = ${filename}`, err); | ||||
| 				}, | ||||
| 			); | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| @@ -460,19 +473,16 @@ export class DriveService { | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		this.registerLogger.debug(`ADD DRIVE FILE: user ${user?.id ?? 'not set'}, name ${detectedName}, tmp ${path}`); | ||||
|  | ||||
| 		//#region Check drive usage | ||||
| 		if (user && !isLink) { | ||||
| 			const usage = await this.driveFileEntityService.calcDriveUsageOf(user); | ||||
| 			const u = await this.usersRepository.findOneBy({ id: user.id }); | ||||
|  | ||||
| 			const instance = await this.metaService.fetch(); | ||||
| 			let driveCapacity = 1024 * 1024 * (this.userEntityService.isLocalUser(user) ? instance.localDriveCapacityMb : instance.remoteDriveCapacityMb); | ||||
|  | ||||
| 			if (this.userEntityService.isLocalUser(user) && u?.driveCapacityOverrideMb != null) { | ||||
| 				driveCapacity = 1024 * 1024 * u.driveCapacityOverrideMb; | ||||
| 				this.registerLogger.debug('drive capacity override applied'); | ||||
| 				this.registerLogger.debug(`overrideCap: ${driveCapacity}bytes, usage: ${usage}bytes, u+s: ${usage + info.size}bytes`); | ||||
| 			} | ||||
| 			const policies = await this.roleService.getUserPolicies(user.id); | ||||
| 			const driveCapacity = 1024 * 1024 * policies.driveCapacityMb; | ||||
| 			this.registerLogger.debug('drive capacity override applied'); | ||||
| 			this.registerLogger.debug(`overrideCap: ${driveCapacity}bytes, usage: ${usage}bytes, u+s: ${usage + info.size}bytes`); | ||||
|  | ||||
| 			this.registerLogger.debug(`drive usage is ${usage} (max: ${driveCapacity})`); | ||||
|  | ||||
|   | ||||
| @@ -34,7 +34,7 @@ export class FederatedInstanceService { | ||||
| 			const i = await this.instancesRepository.insert({ | ||||
| 				id: this.idService.genId(), | ||||
| 				host, | ||||
| 				caughtAt: new Date(), | ||||
| 				firstRetrievedAt: new Date(), | ||||
| 			}).then(x => this.instancesRepository.findOneByOrFail(x.identifiers[0])); | ||||
| 	 | ||||
| 			this.cache.set(host, i); | ||||
|   | ||||
| @@ -1,7 +1,6 @@ | ||||
| import { URL } from 'node:url'; | ||||
| import { Inject, Injectable } from '@nestjs/common'; | ||||
| import { JSDOM } from 'jsdom'; | ||||
| import fetch from 'node-fetch'; | ||||
| import tinycolor from 'tinycolor2'; | ||||
| import type { Instance } from '@/models/entities/Instance.js'; | ||||
| import type { InstancesRepository } from '@/models/index.js'; | ||||
| @@ -191,11 +190,7 @@ export class FetchInstanceMetadataService { | ||||
| 	 | ||||
| 		const faviconUrl = url + '/favicon.ico'; | ||||
| 	 | ||||
| 		const favicon = await fetch(faviconUrl, { | ||||
| 			// TODO | ||||
| 			//timeout: 10000, | ||||
| 			agent: url => this.httpRequestService.getAgentByUrl(url), | ||||
| 		}); | ||||
| 		const favicon = await this.httpRequestService.fetch(faviconUrl, {}, { noOkError: true }); | ||||
| 	 | ||||
| 		if (favicon.ok) { | ||||
| 			return faviconUrl; | ||||
|   | ||||
| @@ -1,67 +1,257 @@ | ||||
| import * as http from 'node:http'; | ||||
| import * as https from 'node:https'; | ||||
| import CacheableLookup from 'cacheable-lookup'; | ||||
| import fetch from 'node-fetch'; | ||||
| import { HttpProxyAgent, HttpsProxyAgent } from 'hpagent'; | ||||
| import { Inject, Injectable } from '@nestjs/common'; | ||||
| import { DI } from '@/di-symbols.js'; | ||||
| import type { Config } from '@/config.js'; | ||||
| import { StatusError } from '@/misc/status-error.js'; | ||||
| import { bindThis } from '@/decorators.js'; | ||||
| import type { Response } from 'node-fetch'; | ||||
| import type { URL } from 'node:url'; | ||||
| import * as undici from 'undici'; | ||||
| import { LookupFunction } from 'node:net'; | ||||
| import { LoggerService } from '@/core/LoggerService.js'; | ||||
| import type Logger from '@/logger.js'; | ||||
|  | ||||
| // true to allow, false to deny | ||||
| export type IpChecker = (ip: string) => boolean; | ||||
|  | ||||
| /*  | ||||
|  *  Child class to create and save Agent for fetch. | ||||
|  *  You should construct this when you want | ||||
|  *  to change timeout, size limit, socket connect function, etc. | ||||
|  */ | ||||
| export class UndiciFetcher { | ||||
| 	/** | ||||
| 	 * Get http non-proxy agent (undici) | ||||
| 	 */ | ||||
| 	public nonProxiedAgent: undici.Agent; | ||||
|  | ||||
| 	/** | ||||
| 	 * Get http proxy or non-proxy agent (undici) | ||||
| 	 */ | ||||
| 	public agent: undici.ProxyAgent | undici.Agent; | ||||
|  | ||||
| 	private proxyBypassHosts: string[]; | ||||
| 	private userAgent: string | undefined; | ||||
|  | ||||
| 	private logger: Logger | undefined; | ||||
|  | ||||
| 	constructor( | ||||
| 		args: { | ||||
| 			agentOptions: undici.Agent.Options; | ||||
| 			proxy?: { | ||||
| 				uri: string; | ||||
| 				options?: undici.Agent.Options; // Override of agentOptions | ||||
| 			}, | ||||
| 			proxyBypassHosts?: string[]; | ||||
| 			userAgent?: string; | ||||
| 		}, | ||||
| 		logger?: Logger, | ||||
| 	) { | ||||
| 		this.logger = logger; | ||||
| 		this.logger?.debug('UndiciFetcher constructor', args); | ||||
|  | ||||
| 		this.proxyBypassHosts = args.proxyBypassHosts ?? []; | ||||
| 		this.userAgent = args.userAgent; | ||||
|  | ||||
| 		this.nonProxiedAgent = new undici.Agent({ | ||||
| 			...args.agentOptions, | ||||
| 			connect: (process.env.NODE_ENV !== 'production' && typeof args.agentOptions.connect !== 'function') | ||||
| 				? (options, cb) => { | ||||
| 					// Custom connector for debug | ||||
| 					undici.buildConnector(args.agentOptions.connect as undici.buildConnector.BuildOptions)(options, (err, socket) => { | ||||
| 						this.logger?.debug('Socket connector called', socket); | ||||
| 						if (err) { | ||||
| 							this.logger?.debug(`Socket error`, err); | ||||
| 							cb(new Error(`Error while socket connecting\n${err}`), null); | ||||
| 							return; | ||||
| 						} | ||||
| 						this.logger?.debug(`Socket connected: port ${socket.localPort} => remote ${socket.remoteAddress}`); | ||||
| 						cb(null, socket); | ||||
| 					}); | ||||
| 				} : args.agentOptions.connect, | ||||
| 		}); | ||||
|  | ||||
| 		this.agent = args.proxy | ||||
| 			? new undici.ProxyAgent({ | ||||
| 				...args.agentOptions, | ||||
| 				...args.proxy.options, | ||||
|  | ||||
| 				uri: args.proxy.uri, | ||||
|  | ||||
| 				connect: (process.env.NODE_ENV !== 'production' && typeof (args.proxy?.options?.connect ?? args.agentOptions.connect) !== 'function') | ||||
| 					? (options, cb) => { | ||||
| 						// Custom connector for debug | ||||
| 						undici.buildConnector((args.proxy?.options?.connect ?? args.agentOptions.connect) as undici.buildConnector.BuildOptions)(options, (err, socket) => { | ||||
| 							this.logger?.debug('Socket connector called (secure)', socket); | ||||
| 							if (err) { | ||||
| 								this.logger?.debug(`Socket error`, err); | ||||
| 								cb(new Error(`Error while socket connecting\n${err}`), null); | ||||
| 								return; | ||||
| 							} | ||||
| 							this.logger?.debug(`Socket connected (secure): port ${socket.localPort} => remote ${socket.remoteAddress}`); | ||||
| 							cb(null, socket); | ||||
| 						}); | ||||
| 					} : (args.proxy?.options?.connect ?? args.agentOptions.connect), | ||||
| 			}) | ||||
| 			: this.nonProxiedAgent; | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * Get agent by URL | ||||
| 	 * @param url URL | ||||
| 	 * @param bypassProxy Allways bypass proxy | ||||
| 	 */ | ||||
| 	@bindThis | ||||
| 	public getAgentByUrl(url: URL, bypassProxy = false): undici.Agent | undici.ProxyAgent { | ||||
| 		if (bypassProxy || this.proxyBypassHosts.includes(url.hostname)) { | ||||
| 			return this.nonProxiedAgent; | ||||
| 		} else { | ||||
| 			return this.agent; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public async fetch( | ||||
| 		url: string | URL, | ||||
| 		options: undici.RequestInit = {}, | ||||
| 		privateOptions: { noOkError?: boolean; bypassProxy?: boolean; } = { noOkError: false, bypassProxy: false } | ||||
| 	): Promise<undici.Response> { | ||||
| 		const res = await undici.fetch(url, { | ||||
| 			dispatcher: this.getAgentByUrl(new URL(url), privateOptions.bypassProxy), | ||||
| 			...options, | ||||
| 			headers: { | ||||
| 				'User-Agent': this.userAgent ?? '', | ||||
| 				...(options.headers ?? {}), | ||||
| 			}, | ||||
| 		}).catch((err) => { | ||||
| 			this.logger?.error(`fetch error to ${typeof url === 'string' ? url : url.href}`, err); | ||||
| 			throw new StatusError('Resource Unreachable', 500, 'Resource Unreachable'); | ||||
| 		}); | ||||
| 		if (!res.ok && !privateOptions.noOkError) { | ||||
| 			throw new StatusError(`${res.status} ${res.statusText}`, res.status, res.statusText); | ||||
| 		} | ||||
| 		return res; | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public async getJson<T extends unknown>(url: string, accept = 'application/json, */*', headers?: Record<string, string>): Promise<T> { | ||||
| 		const res = await this.fetch( | ||||
| 			url, | ||||
| 			{ | ||||
| 				headers: Object.assign({ | ||||
| 					Accept: accept, | ||||
| 				}, headers ?? {}), | ||||
| 			} | ||||
| 		); | ||||
|  | ||||
| 		return await res.json() as T; | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public async getHtml(url: string, accept = 'text/html, */*', headers?: Record<string, string>): Promise<string> { | ||||
| 		const res = await this.fetch( | ||||
| 			url, | ||||
| 			{ | ||||
| 				headers: Object.assign({ | ||||
| 					Accept: accept, | ||||
| 				}, headers ?? {}), | ||||
| 			} | ||||
| 		); | ||||
|  | ||||
| 		return await res.text(); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @Injectable() | ||||
| export class HttpRequestService { | ||||
| 	/** | ||||
| 	 * Get http non-proxy agent | ||||
| 	 */ | ||||
| 	public defaultFetcher: UndiciFetcher; | ||||
| 	public fetch: UndiciFetcher['fetch']; | ||||
| 	public getHtml: UndiciFetcher['getHtml']; | ||||
| 	public defaultJsonFetcher: UndiciFetcher; | ||||
| 	public getJson: UndiciFetcher['getJson']; | ||||
|  | ||||
| 	//#region for old http/https, only used in S3Service | ||||
| 	// http non-proxy agent | ||||
| 	private http: http.Agent; | ||||
|  | ||||
| 	/** | ||||
| 	 * Get https non-proxy agent | ||||
| 	 */ | ||||
| 	// https non-proxy agent | ||||
| 	private https: https.Agent; | ||||
|  | ||||
| 	/** | ||||
| 	 * Get http proxy or non-proxy agent | ||||
| 	 */ | ||||
| 	// http proxy or non-proxy agent | ||||
| 	public httpAgent: http.Agent; | ||||
|  | ||||
| 	/** | ||||
| 	 * Get https proxy or non-proxy agent | ||||
| 	 */ | ||||
| 	// https proxy or non-proxy agent | ||||
| 	public httpsAgent: https.Agent; | ||||
| 	//#endregion | ||||
|  | ||||
| 	public readonly dnsCache: CacheableLookup; | ||||
| 	public readonly clientDefaults: undici.Agent.Options; | ||||
| 	private maxSockets: number; | ||||
|  | ||||
| 	private logger: Logger; | ||||
|  | ||||
| 	constructor( | ||||
| 		@Inject(DI.config) | ||||
| 		private config: Config, | ||||
| 		private loggerService: LoggerService, | ||||
| 	) { | ||||
| 		const cache = new CacheableLookup({ | ||||
| 		this.logger = this.loggerService.getLogger('http-request'); | ||||
|  | ||||
| 		this.dnsCache = new CacheableLookup({ | ||||
| 			maxTtl: 3600,	// 1hours | ||||
| 			errorTtl: 30,	// 30secs | ||||
| 			lookup: false,	// nativeのdns.lookupにfallbackしない | ||||
| 		}); | ||||
| 		 | ||||
|  | ||||
| 		this.clientDefaults = { | ||||
| 			keepAliveTimeout: 30 * 1000, | ||||
| 			keepAliveMaxTimeout: 10 * 60 * 1000, | ||||
| 			keepAliveTimeoutThreshold: 1 * 1000, | ||||
| 			strictContentLength: true, | ||||
| 			headersTimeout: 10 * 1000, | ||||
| 			bodyTimeout: 10 * 1000, | ||||
| 			maxHeaderSize: 16364, // default | ||||
| 			maxResponseSize: 10 * 1024 * 1024, | ||||
| 			maxRedirections: 3, | ||||
| 			connect: { | ||||
| 				timeout: 10 * 1000, // コネクションが確立するまでのタイムアウト | ||||
| 				maxCachedSessions: 300, // TLSセッションのキャッシュ数 https://github.com/nodejs/undici/blob/v5.14.0/lib/core/connect.js#L80 | ||||
| 				lookup: this.dnsCache.lookup as LookupFunction, // https://github.com/nodejs/undici/blob/v5.14.0/lib/core/connect.js#L98 | ||||
| 			}, | ||||
| 		} | ||||
|  | ||||
| 		this.maxSockets = Math.max(64, this.config.deliverJobConcurrency ?? 128); | ||||
|  | ||||
| 		this.defaultFetcher = new UndiciFetcher(this.getStandardUndiciFetcherOption(), this.logger); | ||||
|  | ||||
| 		this.fetch = this.defaultFetcher.fetch; | ||||
| 		this.getHtml = this.defaultFetcher.getHtml; | ||||
|  | ||||
| 		this.defaultJsonFetcher = new UndiciFetcher(this.getStandardUndiciFetcherOption({ | ||||
| 			maxResponseSize: 1024 * 256, | ||||
| 		}), this.logger); | ||||
|  | ||||
| 		this.getJson = this.defaultJsonFetcher.getJson; | ||||
|  | ||||
| 		//#region for old http/https, only used in S3Service | ||||
| 		this.http = new http.Agent({ | ||||
| 			keepAlive: true, | ||||
| 			keepAliveMsecs: 30 * 1000, | ||||
| 			lookup: cache.lookup, | ||||
| 			lookup: this.dnsCache.lookup, | ||||
| 		} as http.AgentOptions); | ||||
| 		 | ||||
| 		this.https = new https.Agent({ | ||||
| 			keepAlive: true, | ||||
| 			keepAliveMsecs: 30 * 1000, | ||||
| 			lookup: cache.lookup, | ||||
| 			lookup: this.dnsCache.lookup, | ||||
| 		} as https.AgentOptions); | ||||
| 		 | ||||
| 		const maxSockets = Math.max(256, config.deliverJobConcurrency ?? 128); | ||||
| 		 | ||||
|  | ||||
| 		this.httpAgent = config.proxy | ||||
| 			? new HttpProxyAgent({ | ||||
| 				keepAlive: true, | ||||
| 				keepAliveMsecs: 30 * 1000, | ||||
| 				maxSockets, | ||||
| 				maxSockets: this.maxSockets, | ||||
| 				maxFreeSockets: 256, | ||||
| 				scheduling: 'lifo', | ||||
| 				proxy: config.proxy, | ||||
| @@ -72,21 +262,42 @@ export class HttpRequestService { | ||||
| 			? new HttpsProxyAgent({ | ||||
| 				keepAlive: true, | ||||
| 				keepAliveMsecs: 30 * 1000, | ||||
| 				maxSockets, | ||||
| 				maxSockets: this.maxSockets, | ||||
| 				maxFreeSockets: 256, | ||||
| 				scheduling: 'lifo', | ||||
| 				proxy: config.proxy, | ||||
| 			}) | ||||
| 			: this.https; | ||||
| 		//#endregion | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public getStandardUndiciFetcherOption(opts: undici.Agent.Options = {}, proxyOpts: undici.Agent.Options = {}) { | ||||
| 		return { | ||||
| 			agentOptions: { | ||||
| 				...this.clientDefaults, | ||||
| 				...opts, | ||||
| 			}, | ||||
| 			...(this.config.proxy ? { | ||||
| 			proxy: { | ||||
| 				uri: this.config.proxy, | ||||
| 				options: { | ||||
| 					connections: this.maxSockets, | ||||
| 					...proxyOpts, | ||||
| 				} | ||||
| 			} | ||||
| 			} : {}), | ||||
| 			userAgent: this.config.userAgent, | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * Get agent by URL | ||||
| 	 * Get http agent by URL | ||||
| 	 * @param url URL | ||||
| 	 * @param bypassProxy Allways bypass proxy | ||||
| 	 */ | ||||
| 	@bindThis | ||||
| 	public getAgentByUrl(url: URL, bypassProxy = false): http.Agent | https.Agent { | ||||
| 	public getHttpAgentByUrl(url: URL, bypassProxy = false): http.Agent | https.Agent { | ||||
| 		if (bypassProxy || (this.config.proxyBypassHosts || []).includes(url.hostname)) { | ||||
| 			return url.protocol === 'http:' ? this.http : this.https; | ||||
| 		} else { | ||||
| @@ -94,67 +305,37 @@ export class HttpRequestService { | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * check ip | ||||
| 	 */ | ||||
| 	@bindThis | ||||
| 	public async getJson(url: string, accept = 'application/json, */*', timeout = 10000, headers?: Record<string, string>): Promise<unknown> { | ||||
| 		const res = await this.getResponse({ | ||||
| 			url, | ||||
| 			method: 'GET', | ||||
| 			headers: Object.assign({ | ||||
| 				'User-Agent': this.config.userAgent, | ||||
| 				Accept: accept, | ||||
| 			}, headers ?? {}), | ||||
| 			timeout, | ||||
| 			size: 1024 * 256, | ||||
| 		}); | ||||
| 	public getConnectorWithIpCheck(connector: undici.buildConnector.connector, checkIp: IpChecker): undici.buildConnector.connectorAsync { | ||||
| 		return (options, cb) => { | ||||
| 			connector(options, (err, socket) => { | ||||
| 				this.logger.debug('Socket connector (with ip checker) called', socket); | ||||
| 				if (err) { | ||||
| 					this.logger.error(`Socket error`, err) | ||||
| 					cb(new Error(`Error while socket connecting\n${err}`), null); | ||||
| 					return; | ||||
| 				} | ||||
|  | ||||
| 		return await res.json(); | ||||
| 	} | ||||
| 				if (socket.remoteAddress == undefined) { | ||||
| 					this.logger.error(`Socket error: remoteAddress is undefined`); | ||||
| 					cb(new Error('remoteAddress is undefined (maybe socket destroyed)'), null); | ||||
| 					return; | ||||
| 				} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public async getHtml(url: string, accept = 'text/html, */*', timeout = 10000, headers?: Record<string, string>): Promise<string> { | ||||
| 		const res = await this.getResponse({ | ||||
| 			url, | ||||
| 			method: 'GET', | ||||
| 			headers: Object.assign({ | ||||
| 				'User-Agent': this.config.userAgent, | ||||
| 				Accept: accept, | ||||
| 			}, headers ?? {}), | ||||
| 			timeout, | ||||
| 		}); | ||||
| 				// allow | ||||
| 				if (checkIp(socket.remoteAddress)) { | ||||
| 					this.logger.debug(`Socket connected (ip ok): ${socket.localPort} => ${socket.remoteAddress}`); | ||||
| 					cb(null, socket); | ||||
| 					return; | ||||
| 				} | ||||
|  | ||||
| 		return await res.text(); | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public async getResponse(args: { | ||||
| 		url: string, | ||||
| 		method: string, | ||||
| 		body?: string, | ||||
| 		headers: Record<string, string>, | ||||
| 		timeout?: number, | ||||
| 		size?: number, | ||||
| 	}): Promise<Response> { | ||||
| 		const timeout = args.timeout ?? 10 * 1000; | ||||
|  | ||||
| 		const controller = new AbortController(); | ||||
| 		setTimeout(() => { | ||||
| 			controller.abort(); | ||||
| 		}, timeout * 6); | ||||
|  | ||||
| 		const res = await fetch(args.url, { | ||||
| 			method: args.method, | ||||
| 			headers: args.headers, | ||||
| 			body: args.body, | ||||
| 			timeout, | ||||
| 			size: args.size ?? 10 * 1024 * 1024, | ||||
| 			agent: (url) => this.getAgentByUrl(url), | ||||
| 			signal: controller.signal, | ||||
| 		}); | ||||
|  | ||||
| 		if (!res.ok) { | ||||
| 			throw new StatusError(`${res.status} ${res.statusText}`, res.status, res.statusText); | ||||
| 		} | ||||
|  | ||||
| 		return res; | ||||
| 				this.logger.error('IP is not allowed', socket); | ||||
| 				cb(new StatusError('IP is not allowed', 403, 'IP is not allowed'), null); | ||||
| 				socket.destroy(); | ||||
| 			}); | ||||
| 		}; | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -4,8 +4,9 @@ import Redis from 'ioredis'; | ||||
| import { DI } from '@/di-symbols.js'; | ||||
| import { Meta } from '@/models/entities/Meta.js'; | ||||
| import { GlobalEventService } from '@/core/GlobalEventService.js'; | ||||
| import type { OnApplicationShutdown } from '@nestjs/common'; | ||||
| import { bindThis } from '@/decorators.js'; | ||||
| import { StreamMessages } from '@/server/api/stream/types.js'; | ||||
| import type { OnApplicationShutdown } from '@nestjs/common'; | ||||
|  | ||||
| @Injectable() | ||||
| export class MetaService implements OnApplicationShutdown { | ||||
| @@ -40,7 +41,7 @@ export class MetaService implements OnApplicationShutdown { | ||||
| 		const obj = JSON.parse(data); | ||||
|  | ||||
| 		if (obj.channel === 'internal') { | ||||
| 			const { type, body } = obj.message; | ||||
| 			const { type, body } = obj.message as StreamMessages['internal']['payload']; | ||||
| 			switch (type) { | ||||
| 				case 'metaUpdated': { | ||||
| 					this.cache = body; | ||||
|   | ||||
| @@ -42,6 +42,7 @@ import { NoteReadService } from '@/core/NoteReadService.js'; | ||||
| import { RemoteUserResolveService } from '@/core/RemoteUserResolveService.js'; | ||||
| import { bindThis } from '@/decorators.js'; | ||||
| import { DB_MAX_NOTE_TEXT_LENGTH } from '@/const.js'; | ||||
| import { RoleService } from '@/core/RoleService.js'; | ||||
|  | ||||
| const mutedWordsCache = new Cache<{ userId: UserProfile['userId']; mutedWords: UserProfile['mutedWords']; }[]>(1000 * 60 * 5); | ||||
|  | ||||
| @@ -186,6 +187,7 @@ export class NoteCreateService { | ||||
| 		private remoteUserResolveService: RemoteUserResolveService, | ||||
| 		private apDeliverManagerService: ApDeliverManagerService, | ||||
| 		private apRendererService: ApRendererService, | ||||
| 		private roleService: RoleService, | ||||
| 		private notesChart: NotesChart, | ||||
| 		private perUserNotesChart: PerUserNotesChart, | ||||
| 		private activeUsersChart: ActiveUsersChart, | ||||
| @@ -197,7 +199,6 @@ export class NoteCreateService { | ||||
| 		id: User['id']; | ||||
| 		username: User['username']; | ||||
| 		host: User['host']; | ||||
| 		isSilenced: User['isSilenced']; | ||||
| 		createdAt: User['createdAt']; | ||||
| 		isBot: User['isBot']; | ||||
| 	}, data: Option, silent = false): Promise<Note> { | ||||
| @@ -224,9 +225,10 @@ export class NoteCreateService { | ||||
| 		if (data.channel != null) data.visibleUsers = []; | ||||
| 		if (data.channel != null) data.localOnly = true; | ||||
|  | ||||
| 		// サイレンス | ||||
| 		if (user.isSilenced && data.visibility === 'public' && data.channel == null) { | ||||
| 			data.visibility = 'home'; | ||||
| 		if (data.visibility === 'public' && data.channel == null) { | ||||
| 			if ((await this.roleService.getUserPolicies(user.id)).canPublicNote === false) { | ||||
| 				data.visibility = 'home'; | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		// Renote対象が「ホームまたは全体」以外の公開範囲ならreject | ||||
| @@ -418,7 +420,6 @@ export class NoteCreateService { | ||||
| 		id: User['id']; | ||||
| 		username: User['username']; | ||||
| 		host: User['host']; | ||||
| 		isSilenced: User['isSilenced']; | ||||
| 		createdAt: User['createdAt']; | ||||
| 		isBot: User['isBot']; | ||||
| 	}, data: Option, silent: boolean, tags: string[], mentionedUsers: MinimumUser[]) { | ||||
|   | ||||
| @@ -12,6 +12,7 @@ import { UserEntityService } from '@/core/entities/UserEntityService.js'; | ||||
| import { ApDeliverManagerService } from '@/core/activitypub/ApDeliverManagerService.js'; | ||||
| import { ApRendererService } from '@/core/activitypub/ApRendererService.js'; | ||||
| import { bindThis } from '@/decorators.js'; | ||||
| import { RoleService } from '@/core/RoleService.js'; | ||||
|  | ||||
| @Injectable() | ||||
| export class NotePiningService { | ||||
| @@ -30,6 +31,7 @@ export class NotePiningService { | ||||
|  | ||||
| 		private userEntityService: UserEntityService, | ||||
| 		private idService: IdService, | ||||
| 		private roleService: RoleService, | ||||
| 		private relayService: RelayService, | ||||
| 		private apDeliverManagerService: ApDeliverManagerService, | ||||
| 		private apRendererService: ApRendererService, | ||||
| @@ -55,7 +57,7 @@ export class NotePiningService { | ||||
|  | ||||
| 		const pinings = await this.userNotePiningsRepository.findBy({ userId: user.id }); | ||||
|  | ||||
| 		if (pinings.length >= 5) { | ||||
| 		if (pinings.length >= (await this.roleService.getUserPolicies(user.id)).pinLimit) { | ||||
| 			throw new IdentifiableError('15a018eb-58e5-4da1-93be-330fcc5e4e1a', 'You can not pin notes any more.'); | ||||
| 		} | ||||
|  | ||||
|   | ||||
							
								
								
									
										298
									
								
								packages/backend/src/core/RoleService.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										298
									
								
								packages/backend/src/core/RoleService.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,298 @@ | ||||
| import { Inject, Injectable } from '@nestjs/common'; | ||||
| import Redis from 'ioredis'; | ||||
| import { In } from 'typeorm'; | ||||
| import type { Role, RoleAssignment, RoleAssignmentsRepository, RolesRepository, UsersRepository } from '@/models/index.js'; | ||||
| import { Cache } from '@/misc/cache.js'; | ||||
| import type { CacheableLocalUser, CacheableUser, ILocalUser, User } from '@/models/entities/User.js'; | ||||
| import { DI } from '@/di-symbols.js'; | ||||
| import { bindThis } from '@/decorators.js'; | ||||
| import { MetaService } from '@/core/MetaService.js'; | ||||
| import { UserCacheService } from '@/core/UserCacheService.js'; | ||||
| import type { RoleCondFormulaValue } from '@/models/entities/Role.js'; | ||||
| import { UserEntityService } from '@/core/entities/UserEntityService.js'; | ||||
| import { StreamMessages } from '@/server/api/stream/types.js'; | ||||
| import type { OnApplicationShutdown } from '@nestjs/common'; | ||||
|  | ||||
| export type RolePolicies = { | ||||
| 	gtlAvailable: boolean; | ||||
| 	ltlAvailable: boolean; | ||||
| 	canPublicNote: boolean; | ||||
| 	canInvite: boolean; | ||||
| 	canManageCustomEmojis: boolean; | ||||
| 	canHideAds: boolean; | ||||
| 	driveCapacityMb: number; | ||||
| 	pinLimit: number; | ||||
| 	antennaLimit: number; | ||||
| 	wordMuteLimit: number; | ||||
| 	webhookLimit: number; | ||||
| 	clipLimit: number; | ||||
| 	noteEachClipsLimit: number; | ||||
| 	userListLimit: number; | ||||
| 	userEachUserListsLimit: number; | ||||
| 	rateLimitFactor: number; | ||||
| }; | ||||
|  | ||||
| export const DEFAULT_POLICIES: RolePolicies = { | ||||
| 	gtlAvailable: true, | ||||
| 	ltlAvailable: true, | ||||
| 	canPublicNote: true, | ||||
| 	canInvite: false, | ||||
| 	canManageCustomEmojis: false, | ||||
| 	canHideAds: false, | ||||
| 	driveCapacityMb: 100, | ||||
| 	pinLimit: 5, | ||||
| 	antennaLimit: 5, | ||||
| 	wordMuteLimit: 200, | ||||
| 	webhookLimit: 3, | ||||
| 	clipLimit: 10, | ||||
| 	noteEachClipsLimit: 200, | ||||
| 	userListLimit: 10, | ||||
| 	userEachUserListsLimit: 50, | ||||
| 	rateLimitFactor: 1, | ||||
| }; | ||||
|  | ||||
| @Injectable() | ||||
| export class RoleService implements OnApplicationShutdown { | ||||
| 	private rolesCache: Cache<Role[]>; | ||||
| 	private roleAssignmentByUserIdCache: Cache<RoleAssignment[]>; | ||||
|  | ||||
| 	constructor( | ||||
| 		@Inject(DI.redisSubscriber) | ||||
| 		private redisSubscriber: Redis.Redis, | ||||
|  | ||||
| 		@Inject(DI.usersRepository) | ||||
| 		private usersRepository: UsersRepository, | ||||
|  | ||||
| 		@Inject(DI.rolesRepository) | ||||
| 		private rolesRepository: RolesRepository, | ||||
|  | ||||
| 		@Inject(DI.roleAssignmentsRepository) | ||||
| 		private roleAssignmentsRepository: RoleAssignmentsRepository, | ||||
|  | ||||
| 		private metaService: MetaService, | ||||
| 		private userCacheService: UserCacheService, | ||||
| 		private userEntityService: UserEntityService, | ||||
| 	) { | ||||
| 		//this.onMessage = this.onMessage.bind(this); | ||||
|  | ||||
| 		this.rolesCache = new Cache<Role[]>(Infinity); | ||||
| 		this.roleAssignmentByUserIdCache = new Cache<RoleAssignment[]>(Infinity); | ||||
|  | ||||
| 		this.redisSubscriber.on('message', this.onMessage); | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	private async onMessage(_: string, data: string): Promise<void> { | ||||
| 		const obj = JSON.parse(data); | ||||
|  | ||||
| 		if (obj.channel === 'internal') { | ||||
| 			const { type, body } = obj.message as StreamMessages['internal']['payload']; | ||||
| 			switch (type) { | ||||
| 				case 'roleCreated': { | ||||
| 					const cached = this.rolesCache.get(null); | ||||
| 					if (cached) { | ||||
| 						body.createdAt = new Date(body.createdAt); | ||||
| 						body.updatedAt = new Date(body.updatedAt); | ||||
| 						body.lastUsedAt = new Date(body.lastUsedAt); | ||||
| 						cached.push(body); | ||||
| 					} | ||||
| 					break; | ||||
| 				} | ||||
| 				case 'roleUpdated': { | ||||
| 					const cached = this.rolesCache.get(null); | ||||
| 					if (cached) { | ||||
| 						const i = cached.findIndex(x => x.id === body.id); | ||||
| 						if (i > -1) { | ||||
| 							body.createdAt = new Date(body.createdAt); | ||||
| 							body.updatedAt = new Date(body.updatedAt); | ||||
| 							body.lastUsedAt = new Date(body.lastUsedAt); | ||||
| 							cached[i] = body; | ||||
| 						} | ||||
| 					} | ||||
| 					break; | ||||
| 				} | ||||
| 				case 'roleDeleted': { | ||||
| 					const cached = this.rolesCache.get(null); | ||||
| 					if (cached) { | ||||
| 						this.rolesCache.set(null, cached.filter(x => x.id !== body.id)); | ||||
| 					} | ||||
| 					break; | ||||
| 				} | ||||
| 				case 'userRoleAssigned': { | ||||
| 					const cached = this.roleAssignmentByUserIdCache.get(body.userId); | ||||
| 					if (cached) { | ||||
| 						body.createdAt = new Date(body.createdAt); | ||||
| 						cached.push(body); | ||||
| 					} | ||||
| 					break; | ||||
| 				} | ||||
| 				case 'userRoleUnassigned': { | ||||
| 					const cached = this.roleAssignmentByUserIdCache.get(body.userId); | ||||
| 					if (cached) { | ||||
| 						this.roleAssignmentByUserIdCache.set(body.userId, cached.filter(x => x.id !== body.id)); | ||||
| 					} | ||||
| 					break; | ||||
| 				} | ||||
| 				default: | ||||
| 					break; | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	private evalCond(user: User, value: RoleCondFormulaValue): boolean { | ||||
| 		try { | ||||
| 			switch (value.type) { | ||||
| 				case 'and': { | ||||
| 					return value.values.every(v => this.evalCond(user, v)); | ||||
| 				} | ||||
| 				case 'or': { | ||||
| 					return value.values.some(v => this.evalCond(user, v)); | ||||
| 				} | ||||
| 				case 'not': { | ||||
| 					return !this.evalCond(user, value.value); | ||||
| 				} | ||||
| 				case 'isLocal': { | ||||
| 					return this.userEntityService.isLocalUser(user); | ||||
| 				} | ||||
| 				case 'isRemote': { | ||||
| 					return this.userEntityService.isRemoteUser(user); | ||||
| 				} | ||||
| 				case 'createdLessThan': { | ||||
| 					return user.createdAt.getTime() > (Date.now() - (value.sec * 1000)); | ||||
| 				} | ||||
| 				case 'createdMoreThan': { | ||||
| 					return user.createdAt.getTime() < (Date.now() - (value.sec * 1000)); | ||||
| 				} | ||||
| 				case 'followersLessThanOrEq': { | ||||
| 					return user.followersCount <= value.value; | ||||
| 				} | ||||
| 				case 'followersMoreThanOrEq': { | ||||
| 					return user.followersCount >= value.value; | ||||
| 				} | ||||
| 				case 'followingLessThanOrEq': { | ||||
| 					return user.followingCount <= value.value; | ||||
| 				} | ||||
| 				case 'followingMoreThanOrEq': { | ||||
| 					return user.followingCount >= value.value; | ||||
| 				} | ||||
| 				default: | ||||
| 					return false; | ||||
| 			} | ||||
| 		} catch (err) { | ||||
| 			// TODO: log error | ||||
| 			return false; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public async getUserRoles(userId: User['id']) { | ||||
| 		const assigns = await this.roleAssignmentByUserIdCache.fetch(userId, () => this.roleAssignmentsRepository.findBy({ userId })); | ||||
| 		const assignedRoleIds = assigns.map(x => x.roleId); | ||||
| 		const roles = await this.rolesCache.fetch(null, () => this.rolesRepository.findBy({})); | ||||
| 		const assignedRoles = roles.filter(r => assignedRoleIds.includes(r.id)); | ||||
| 		const user = roles.some(r => r.target === 'conditional') ? await this.userCacheService.findById(userId) : null; | ||||
| 		const matchedCondRoles = roles.filter(r => r.target === 'conditional' && this.evalCond(user!, r.condFormula)); | ||||
| 		return [...assignedRoles, ...matchedCondRoles]; | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public async getUserPolicies(userId: User['id'] | null): Promise<RolePolicies> { | ||||
| 		const meta = await this.metaService.fetch(); | ||||
| 		const basePolicies = { ...DEFAULT_POLICIES, ...meta.policies }; | ||||
|  | ||||
| 		if (userId == null) return basePolicies; | ||||
|  | ||||
| 		const roles = await this.getUserRoles(userId); | ||||
|  | ||||
| 		function calc<T extends keyof RolePolicies>(name: T, aggregate: (values: RolePolicies[T][]) => RolePolicies[T]) { | ||||
| 			if (roles.length === 0) return basePolicies[name]; | ||||
|  | ||||
| 			const policies = roles.map(role => role.policies[name] ?? { priority: 0, useDefault: true }); | ||||
|  | ||||
| 			const p2 = policies.filter(policy => policy.priority === 2); | ||||
| 			if (p2.length > 0) return aggregate(p2.map(policy => policy.useDefault ? basePolicies[name] : policy.value)); | ||||
|  | ||||
| 			const p1 = policies.filter(policy => policy.priority === 1); | ||||
| 			if (p1.length > 0) return aggregate(p1.map(policy => policy.useDefault ? basePolicies[name] : policy.value)); | ||||
|  | ||||
| 			return aggregate(policies.map(policy => policy.useDefault ? basePolicies[name] : policy.value)); | ||||
| 		} | ||||
|  | ||||
| 		return { | ||||
| 			gtlAvailable: calc('gtlAvailable', vs => vs.some(v => v === true)), | ||||
| 			ltlAvailable: calc('ltlAvailable', vs => vs.some(v => v === true)), | ||||
| 			canPublicNote: calc('canPublicNote', vs => vs.some(v => v === true)), | ||||
| 			canInvite: calc('canInvite', vs => vs.some(v => v === true)), | ||||
| 			canManageCustomEmojis: calc('canManageCustomEmojis', vs => vs.some(v => v === true)), | ||||
| 			canHideAds: calc('canHideAds', vs => vs.some(v => v === true)), | ||||
| 			driveCapacityMb: calc('driveCapacityMb', vs => Math.max(...vs)), | ||||
| 			pinLimit: calc('pinLimit', vs => Math.max(...vs)), | ||||
| 			antennaLimit: calc('antennaLimit', vs => Math.max(...vs)), | ||||
| 			wordMuteLimit: calc('wordMuteLimit', vs => Math.max(...vs)), | ||||
| 			webhookLimit: calc('webhookLimit', vs => Math.max(...vs)), | ||||
| 			clipLimit: calc('clipLimit', vs => Math.max(...vs)), | ||||
| 			noteEachClipsLimit: calc('noteEachClipsLimit', vs => Math.max(...vs)), | ||||
| 			userListLimit: calc('userListLimit', vs => Math.max(...vs)), | ||||
| 			userEachUserListsLimit: calc('userEachUserListsLimit', vs => Math.max(...vs)), | ||||
| 			rateLimitFactor: calc('rateLimitFactor', vs => Math.max(...vs)), | ||||
| 		}; | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public async isModerator(user: { id: User['id']; isRoot: User['isRoot'] } | null): Promise<boolean> { | ||||
| 		if (user == null) return false; | ||||
| 		return user.isRoot || (await this.getUserRoles(user.id)).some(r => r.isModerator || r.isAdministrator); | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public async isAdministrator(user: { id: User['id']; isRoot: User['isRoot'] } | null): Promise<boolean> { | ||||
| 		if (user == null) return false; | ||||
| 		return user.isRoot || (await this.getUserRoles(user.id)).some(r => r.isAdministrator); | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public async getModeratorIds(includeAdmins = true): Promise<User['id'][]> { | ||||
| 		const roles = await this.rolesCache.fetch(null, () => this.rolesRepository.findBy({})); | ||||
| 		const moderatorRoles = includeAdmins ? roles.filter(r => r.isModerator || r.isAdministrator) : roles.filter(r => r.isModerator); | ||||
| 		const assigns = moderatorRoles.length > 0 ? await this.roleAssignmentsRepository.findBy({ | ||||
| 			roleId: In(moderatorRoles.map(r => r.id)), | ||||
| 		}) : []; | ||||
| 		// TODO: isRootなアカウントも含める | ||||
| 		return assigns.map(a => a.userId); | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public async getModerators(includeAdmins = true): Promise<User[]> { | ||||
| 		const ids = await this.getModeratorIds(includeAdmins); | ||||
| 		const users = ids.length > 0 ? await this.usersRepository.findBy({ | ||||
| 			id: In(ids), | ||||
| 		}) : []; | ||||
| 		return users; | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public async getAdministratorIds(): Promise<User['id'][]> { | ||||
| 		const roles = await this.rolesCache.fetch(null, () => this.rolesRepository.findBy({})); | ||||
| 		const administratorRoles = roles.filter(r => r.isAdministrator); | ||||
| 		const assigns = administratorRoles.length > 0 ? await this.roleAssignmentsRepository.findBy({ | ||||
| 			roleId: In(administratorRoles.map(r => r.id)), | ||||
| 		}) : []; | ||||
| 		// TODO: isRootなアカウントも含める | ||||
| 		return assigns.map(a => a.userId); | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public async getAdministrators(): Promise<User[]> { | ||||
| 		const ids = await this.getAdministratorIds(); | ||||
| 		const users = ids.length > 0 ? await this.usersRepository.findBy({ | ||||
| 			id: In(ids), | ||||
| 		}) : []; | ||||
| 		return users; | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public onApplicationShutdown(signal?: string | undefined) { | ||||
| 		this.redisSubscriber.off('message', this.onMessage); | ||||
| 	} | ||||
| } | ||||
| @@ -33,7 +33,7 @@ export class S3Service { | ||||
| 				? false | ||||
| 				: meta.objectStorageS3ForcePathStyle, | ||||
| 			httpOptions: { | ||||
| 				agent: this.httpRequestService.getAgentByUrl(new URL(u), !meta.objectStorageUseProxy), | ||||
| 				agent: this.httpRequestService.getHttpAgentByUrl(new URL(u), !meta.objectStorageUseProxy), | ||||
| 			}, | ||||
| 		}); | ||||
| 	} | ||||
|   | ||||
| @@ -11,10 +11,10 @@ import { IdService } from '@/core/IdService.js'; | ||||
| import { UserKeypair } from '@/models/entities/UserKeypair.js'; | ||||
| import { UsedUsername } from '@/models/entities/UsedUsername.js'; | ||||
| import generateUserToken from '@/misc/generate-native-user-token.js'; | ||||
| import UsersChart from './chart/charts/users.js'; | ||||
| import { UserEntityService } from '@/core/entities/UserEntityService.js'; | ||||
| import { UtilityService } from './UtilityService.js'; | ||||
| import { bindThis } from '@/decorators.js'; | ||||
| import UsersChart from './chart/charts/users.js'; | ||||
| import { UtilityService } from './UtilityService.js'; | ||||
|  | ||||
| @Injectable() | ||||
| export class SignupService { | ||||
| @@ -112,7 +112,7 @@ export class SignupService { | ||||
| 				usernameLower: username.toLowerCase(), | ||||
| 				host: this.utilityService.toPunyNullable(host), | ||||
| 				token: secret, | ||||
| 				isAdmin: (await this.usersRepository.countBy({ | ||||
| 				isRoot: (await this.usersRepository.countBy({ | ||||
| 					host: IsNull(), | ||||
| 				})) === 0, | ||||
| 			})); | ||||
|   | ||||
| @@ -2,11 +2,12 @@ import { Inject, Injectable } from '@nestjs/common'; | ||||
| import Redis from 'ioredis'; | ||||
| import type { UsersRepository } from '@/models/index.js'; | ||||
| import { Cache } from '@/misc/cache.js'; | ||||
| import type { CacheableLocalUser, CacheableUser, ILocalUser } from '@/models/entities/User.js'; | ||||
| import type { CacheableLocalUser, CacheableUser, ILocalUser, User } from '@/models/entities/User.js'; | ||||
| import { DI } from '@/di-symbols.js'; | ||||
| import { UserEntityService } from '@/core/entities/UserEntityService.js'; | ||||
| import type { OnApplicationShutdown } from '@nestjs/common'; | ||||
| import { bindThis } from '@/decorators.js'; | ||||
| import { StreamMessages } from '@/server/api/stream/types.js'; | ||||
| import type { OnApplicationShutdown } from '@nestjs/common'; | ||||
|  | ||||
| @Injectable() | ||||
| export class UserCacheService implements OnApplicationShutdown { | ||||
| @@ -39,11 +40,9 @@ export class UserCacheService implements OnApplicationShutdown { | ||||
| 		const obj = JSON.parse(data); | ||||
|  | ||||
| 		if (obj.channel === 'internal') { | ||||
| 			const { type, body } = obj.message; | ||||
| 			const { type, body } = obj.message as StreamMessages['internal']['payload']; | ||||
| 			switch (type) { | ||||
| 				case 'userChangeSuspendedState': | ||||
| 				case 'userChangeSilencedState': | ||||
| 				case 'userChangeModeratorState': | ||||
| 				case 'remoteUserUpdated': { | ||||
| 					const user = await this.usersRepository.findOneByOrFail({ id: body.id }); | ||||
| 					this.userByIdCache.set(user.id, user); | ||||
| @@ -64,12 +63,24 @@ export class UserCacheService implements OnApplicationShutdown { | ||||
| 					this.localUserByNativeTokenCache.set(body.newToken, user); | ||||
| 					break; | ||||
| 				} | ||||
| 				case 'follow': { | ||||
| 					const follower = this.userByIdCache.get(body.followerId); | ||||
| 					if (follower) follower.followingCount++; | ||||
| 					const followee = this.userByIdCache.get(body.followeeId); | ||||
| 					if (followee) followee.followersCount++; | ||||
| 					break; | ||||
| 				} | ||||
| 				default: | ||||
| 					break; | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public findById(userId: User['id']) { | ||||
| 		return this.userByIdCache.fetch(userId, () => this.usersRepository.findOneByOrFail({ id: userId })); | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public onApplicationShutdown(signal?: string | undefined) { | ||||
| 		this.redisSubscriber.off('message', this.onMessage); | ||||
|   | ||||
| @@ -62,6 +62,7 @@ export class UserFollowingService { | ||||
| 		private federatedInstanceService: FederatedInstanceService, | ||||
| 		private webhookService: WebhookService, | ||||
| 		private apRendererService: ApRendererService, | ||||
| 		private globalEventService: GlobalEventService, | ||||
| 		private perUserFollowingChart: PerUserFollowingChart, | ||||
| 		private instanceChart: InstanceChart, | ||||
| 	) { | ||||
| @@ -195,6 +196,8 @@ export class UserFollowingService { | ||||
| 		} | ||||
| 	 | ||||
| 		if (alreadyFollowed) return; | ||||
|  | ||||
| 		this.globalEventService.publishInternalEvent('follow', { followerId: follower.id, followeeId: followee.id }); | ||||
| 	 | ||||
| 		//#region Increment counts | ||||
| 		await Promise.all([ | ||||
| @@ -314,6 +317,8 @@ export class UserFollowingService { | ||||
| 		follower: {id: User['id']; host: User['host']; }, | ||||
| 		followee: { id: User['id']; host: User['host']; }, | ||||
| 	): Promise<void> { | ||||
| 		this.globalEventService.publishInternalEvent('unfollow', { followerId: follower.id, followeeId: followee.id }); | ||||
| 	 | ||||
| 		//#region Decrement following / followers counts | ||||
| 		await Promise.all([ | ||||
| 			this.usersRepository.decrement({ id: follower.id }, 'followingCount', 1), | ||||
|   | ||||
| @@ -10,6 +10,7 @@ import { DI } from '@/di-symbols.js'; | ||||
| import { UserEntityService } from '@/core/entities/UserEntityService.js'; | ||||
| import { ProxyAccountService } from '@/core/ProxyAccountService.js'; | ||||
| import { bindThis } from '@/decorators.js'; | ||||
| import { RoleService } from '@/core/RoleService.js'; | ||||
|  | ||||
| @Injectable() | ||||
| export class UserListService { | ||||
| @@ -23,13 +24,21 @@ export class UserListService { | ||||
| 		private userEntityService: UserEntityService, | ||||
| 		private idService: IdService, | ||||
| 		private userFollowingService: UserFollowingService, | ||||
| 		private roleService: RoleService, | ||||
| 		private globalEventServie: GlobalEventService, | ||||
| 		private proxyAccountService: ProxyAccountService, | ||||
| 	) { | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public async push(target: User, list: UserList) { | ||||
| 	public async push(target: User, list: UserList, me: User) { | ||||
| 		const currentCount = await this.userListJoiningsRepository.countBy({ | ||||
| 			userListId: list.id, | ||||
| 		}); | ||||
| 		if (currentCount > (await this.roleService.getUserPolicies(me.id)).userEachUserListsLimit) { | ||||
| 			throw new Error('Too many users'); | ||||
| 		} | ||||
|  | ||||
| 		await this.userListJoiningsRepository.insert({ | ||||
| 			id: this.idService.genId(), | ||||
| 			createdAt: new Date(), | ||||
|   | ||||
| @@ -24,6 +24,12 @@ export class UtilityService { | ||||
| 		return this.toPuny(this.config.host) === this.toPuny(host); | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public isBlockedHost(blockedHosts: string[], host: string | null): boolean { | ||||
| 		if (host == null) return false; | ||||
| 		return blockedHosts.some(x => `.${host.toLowerCase()}`.endsWith(`.${x}`)); | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public extractDbHost(uri: string): string { | ||||
| 		const url = new URL(uri); | ||||
|   | ||||
| @@ -30,7 +30,7 @@ export class WebfingerService { | ||||
| 	public async webfinger(query: string): Promise<IWebFinger> { | ||||
| 		const url = this.genUrl(query); | ||||
|  | ||||
| 		return await this.httpRequestService.getJson(url, 'application/jrd+json, application/json') as IWebFinger; | ||||
| 		return await this.httpRequestService.getJson<IWebFinger>(url, 'application/jrd+json, application/json'); | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
|   | ||||
| @@ -3,8 +3,9 @@ import Redis from 'ioredis'; | ||||
| import type { WebhooksRepository } from '@/models/index.js'; | ||||
| import type { Webhook } from '@/models/entities/Webhook.js'; | ||||
| import { DI } from '@/di-symbols.js'; | ||||
| import type { OnApplicationShutdown } from '@nestjs/common'; | ||||
| import { bindThis } from '@/decorators.js'; | ||||
| import { StreamMessages } from '@/server/api/stream/types.js'; | ||||
| import type { OnApplicationShutdown } from '@nestjs/common'; | ||||
|  | ||||
| @Injectable() | ||||
| export class WebhookService implements OnApplicationShutdown { | ||||
| @@ -39,7 +40,7 @@ export class WebhookService implements OnApplicationShutdown { | ||||
| 		const obj = JSON.parse(data); | ||||
|  | ||||
| 		if (obj.channel === 'internal') { | ||||
| 			const { type, body } = obj.message; | ||||
| 			const { type, body } = obj.message as StreamMessages['internal']['payload']; | ||||
| 			switch (type) { | ||||
| 				case 'webhookCreated': | ||||
| 					if (body.active) { | ||||
|   | ||||
| @@ -159,7 +159,7 @@ export class ApDbResolverService { | ||||
| 		if (key == null) return null; | ||||
|  | ||||
| 		return { | ||||
| 			user: await this.userCacheService.userByIdCache.fetch(key.userId, () => this.usersRepository.findOneByOrFail({ id: key.userId })) as CacheableRemoteUser, | ||||
| 			user: await this.userCacheService.findById(key.userId) as CacheableRemoteUser, | ||||
| 			key, | ||||
| 		}; | ||||
| 	} | ||||
|   | ||||
| @@ -291,7 +291,7 @@ export class ApInboxService { | ||||
|  | ||||
| 		// アナウンス先をブロックしてたら中断 | ||||
| 		const meta = await this.metaService.fetch(); | ||||
| 		if (meta.blockedHosts.includes(this.utilityService.extractDbHost(uri))) return; | ||||
| 		if (this.utilityService.isBlockedHost(meta.blockedHosts, this.utilityService.extractDbHost(uri))) return; | ||||
|  | ||||
| 		const unlock = await this.appLockService.getApLock(uri); | ||||
|  | ||||
|   | ||||
| @@ -5,8 +5,10 @@ import { DI } from '@/di-symbols.js'; | ||||
| import type { Config } from '@/config.js'; | ||||
| import type { User } from '@/models/entities/User.js'; | ||||
| import { UserKeypairStoreService } from '@/core/UserKeypairStoreService.js'; | ||||
| import { HttpRequestService } from '@/core/HttpRequestService.js'; | ||||
| import { HttpRequestService, UndiciFetcher } from '@/core/HttpRequestService.js'; | ||||
| import { LoggerService } from '@/core/LoggerService.js'; | ||||
| import { bindThis } from '@/decorators.js'; | ||||
| import type Logger from '@/logger.js'; | ||||
|  | ||||
| type Request = { | ||||
| 	url: string; | ||||
| @@ -28,13 +30,21 @@ type PrivateKey = { | ||||
|  | ||||
| @Injectable() | ||||
| export class ApRequestService { | ||||
| 	private undiciFetcher: UndiciFetcher; | ||||
| 	private logger: Logger; | ||||
|  | ||||
| 	constructor( | ||||
| 		@Inject(DI.config) | ||||
| 		private config: Config, | ||||
|  | ||||
| 		private userKeypairStoreService: UserKeypairStoreService, | ||||
| 		private httpRequestService: HttpRequestService, | ||||
| 		private loggerService: LoggerService, | ||||
| 	) { | ||||
| 		this.logger = this.loggerService?.getLogger('ap-request'); // なぜか TypeError: Cannot read properties of undefined (reading 'getLogger') と言われる | ||||
| 		this.undiciFetcher = new UndiciFetcher(this.httpRequestService.getStandardUndiciFetcherOption({ | ||||
| 			maxRedirections: 0, | ||||
| 		}), this.logger ); | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| @@ -47,7 +57,7 @@ export class ApRequestService { | ||||
| 			method: 'POST', | ||||
| 			headers: this.objectAssignWithLcKey({ | ||||
| 				'Date': new Date().toUTCString(), | ||||
| 				'Host': u.hostname, | ||||
| 				'Host': u.host, | ||||
| 				'Content-Type': 'application/activity+json', | ||||
| 				'Digest': digestHeader, | ||||
| 			}, args.additionalHeaders), | ||||
| @@ -73,7 +83,7 @@ export class ApRequestService { | ||||
| 			headers: this.objectAssignWithLcKey({ | ||||
| 				'Accept': 'application/activity+json, application/ld+json', | ||||
| 				'Date': new Date().toUTCString(), | ||||
| 				'Host': new URL(args.url).hostname, | ||||
| 				'Host': new URL(args.url).host, | ||||
| 			}, args.additionalHeaders), | ||||
| 		}; | ||||
|  | ||||
| @@ -96,6 +106,8 @@ export class ApRequestService { | ||||
| 		request.headers = this.objectAssignWithLcKey(request.headers, { | ||||
| 			Signature: signatureHeader, | ||||
| 		}); | ||||
| 		// node-fetch will generate this for us. if we keep 'Host', it won't change with redirects! | ||||
| 		delete request.headers['host']; | ||||
|  | ||||
| 		return { | ||||
| 			request, | ||||
| @@ -148,16 +160,17 @@ export class ApRequestService { | ||||
| 			url, | ||||
| 			body, | ||||
| 			additionalHeaders: { | ||||
| 				'User-Agent': this.config.userAgent, | ||||
| 			}, | ||||
| 		}); | ||||
|  | ||||
| 		await this.httpRequestService.getResponse({ | ||||
| 		await this.undiciFetcher.fetch( | ||||
| 			url, | ||||
| 			method: req.request.method, | ||||
| 			headers: req.request.headers, | ||||
| 			body, | ||||
| 		}); | ||||
| 			{ | ||||
| 				method: req.request.method, | ||||
| 				headers: req.request.headers, | ||||
| 				body, | ||||
| 			} | ||||
| 		); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| @@ -176,15 +189,16 @@ export class ApRequestService { | ||||
| 			}, | ||||
| 			url, | ||||
| 			additionalHeaders: { | ||||
| 				'User-Agent': this.config.userAgent, | ||||
| 			}, | ||||
| 		}); | ||||
|  | ||||
| 		const res = await this.httpRequestService.getResponse({ | ||||
| 		const res = await this.httpRequestService.fetch( | ||||
| 			url, | ||||
| 			method: req.request.method, | ||||
| 			headers: req.request.headers, | ||||
| 		}); | ||||
| 			{ | ||||
| 				method: req.request.method, | ||||
| 				headers: req.request.headers, | ||||
| 			} | ||||
| 		); | ||||
|  | ||||
| 		return await res.json(); | ||||
| 	} | ||||
|   | ||||
| @@ -4,7 +4,7 @@ import { InstanceActorService } from '@/core/InstanceActorService.js'; | ||||
| import type { NotesRepository, PollsRepository, NoteReactionsRepository, UsersRepository } from '@/models/index.js'; | ||||
| import type { Config } from '@/config.js'; | ||||
| import { MetaService } from '@/core/MetaService.js'; | ||||
| import { HttpRequestService } from '@/core/HttpRequestService.js'; | ||||
| import { HttpRequestService, UndiciFetcher } from '@/core/HttpRequestService.js'; | ||||
| import { DI } from '@/di-symbols.js'; | ||||
| import { UtilityService } from '@/core/UtilityService.js'; | ||||
| import { bindThis } from '@/decorators.js'; | ||||
| @@ -12,11 +12,15 @@ import { isCollectionOrOrderedCollection } from './type.js'; | ||||
| import { ApDbResolverService } from './ApDbResolverService.js'; | ||||
| import { ApRendererService } from './ApRendererService.js'; | ||||
| import { ApRequestService } from './ApRequestService.js'; | ||||
| import { LoggerService } from '@/core/LoggerService.js'; | ||||
| import type { IObject, ICollection, IOrderedCollection } from './type.js'; | ||||
| import type Logger from '@/logger.js'; | ||||
|  | ||||
| export class Resolver { | ||||
| 	private history: Set<string>; | ||||
| 	private user?: ILocalUser; | ||||
| 	private undiciFetcher: UndiciFetcher; | ||||
| 	private logger: Logger; | ||||
|  | ||||
| 	constructor( | ||||
| 		private config: Config, | ||||
| @@ -31,9 +35,14 @@ export class Resolver { | ||||
| 		private httpRequestService: HttpRequestService, | ||||
| 		private apRendererService: ApRendererService, | ||||
| 		private apDbResolverService: ApDbResolverService, | ||||
| 		private loggerService: LoggerService, | ||||
| 		private recursionLimit = 100, | ||||
| 	) { | ||||
| 		this.history = new Set(); | ||||
| 		this.logger = this.loggerService?.getLogger('ap-resolve');  // なぜか TypeError: Cannot read properties of undefined (reading 'getLogger') と言われる | ||||
| 		this.undiciFetcher = new UndiciFetcher(this.httpRequestService.getStandardUndiciFetcherOption({ | ||||
| 			maxRedirections: 0, | ||||
| 		}), this.logger); | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| @@ -87,7 +96,7 @@ export class Resolver { | ||||
| 		} | ||||
|  | ||||
| 		const meta = await this.metaService.fetch(); | ||||
| 		if (meta.blockedHosts.includes(host)) { | ||||
| 		if (this.utilityService.isBlockedHost(meta.blockedHosts, host)) { | ||||
| 			throw new Error('Instance is blocked'); | ||||
| 		} | ||||
|  | ||||
| @@ -96,8 +105,8 @@ export class Resolver { | ||||
| 		} | ||||
|  | ||||
| 		const object = (this.user | ||||
| 			? await this.apRequestService.signedGet(value, this.user) | ||||
| 			: await this.httpRequestService.getJson(value, 'application/activity+json, application/ld+json')) as IObject; | ||||
| 			? await this.apRequestService.signedGet(value, this.user) as IObject | ||||
| 			: await this.undiciFetcher.getJson<IObject>(value, 'application/activity+json, application/ld+json')); | ||||
|  | ||||
| 		if (object == null || ( | ||||
| 			Array.isArray(object['@context']) ? | ||||
|   | ||||
| @@ -1,6 +1,5 @@ | ||||
| import * as crypto from 'node:crypto'; | ||||
| import { Inject, Injectable } from '@nestjs/common'; | ||||
| import fetch from 'node-fetch'; | ||||
| import { HttpRequestService } from '@/core/HttpRequestService.js'; | ||||
| import { bindThis } from '@/decorators.js'; | ||||
| import { CONTEXTS } from './misc/contexts.js'; | ||||
| @@ -116,14 +115,19 @@ class LdSignature { | ||||
|  | ||||
| 	@bindThis | ||||
| 	private async fetchDocument(url: string) { | ||||
| 		const json = await fetch(url, { | ||||
| 			headers: { | ||||
| 				Accept: 'application/ld+json, application/json', | ||||
| 		const json = await this.httpRequestService.fetch( | ||||
| 			url, | ||||
| 			{ | ||||
| 				headers: { | ||||
| 					Accept: 'application/ld+json, application/json', | ||||
| 				}, | ||||
| 				// TODO | ||||
| 				//timeout: this.loderTimeout, | ||||
| 			}, | ||||
| 			// TODO | ||||
| 			//timeout: this.loderTimeout, | ||||
| 			agent: u => u.protocol === 'http:' ? this.httpRequestService.httpAgent : this.httpRequestService.httpsAgent, | ||||
| 		}).then(res => { | ||||
| 			{ | ||||
| 				noOkError: true, | ||||
| 			} | ||||
| 		).then(res => { | ||||
| 			if (!res.ok) { | ||||
| 				throw `${res.status} ${res.statusText}`; | ||||
| 			} else { | ||||
|   | ||||
| @@ -324,7 +324,7 @@ export class ApNoteService { | ||||
| 	 | ||||
| 		// ブロックしてたら中断 | ||||
| 		const meta = await this.metaService.fetch(); | ||||
| 		if (meta.blockedHosts.includes(this.utilityService.extractDbHost(uri))) throw { statusCode: 451 }; | ||||
| 		if (this.utilityService.isBlockedHost(meta.blockedHosts, this.utilityService.extractDbHost(uri))) throw { statusCode: 451 }; | ||||
| 	 | ||||
| 		const unlock = await this.appLockService.getApLock(uri); | ||||
| 	 | ||||
|   | ||||
| @@ -54,7 +54,7 @@ export class ChartManagementService implements OnApplicationShutdown { | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public async run() { | ||||
| 	public async start() { | ||||
| 		// 20分おきにメモリ情報をDBに書き込み | ||||
| 		this.saveIntervalId = setInterval(() => { | ||||
| 			for (const chart of this.charts) { | ||||
|   | ||||
| @@ -61,21 +61,21 @@ export default class FederationChart extends Chart<typeof schema> { | ||||
| 			this.followingsRepository.createQueryBuilder('following') | ||||
| 				.select('COUNT(DISTINCT following.followeeHost)') | ||||
| 				.where('following.followeeHost IS NOT NULL') | ||||
| 				.andWhere(meta.blockedHosts.length === 0 ? '1=1' : 'following.followeeHost NOT IN (:...blocked)', { blocked: meta.blockedHosts }) | ||||
| 				.andWhere(meta.blockedHosts.length === 0 ? '1=1' : 'following.followeeHost NOT ILIKE ANY(ARRAY[:...blocked])', { blocked: meta.blockedHosts.flatMap(x => [x, `%.${x}`]) }) | ||||
| 				.andWhere(`following.followeeHost NOT IN (${ suspendedInstancesQuery.getQuery() })`) | ||||
| 				.getRawOne() | ||||
| 				.then(x => parseInt(x.count, 10)), | ||||
| 			this.followingsRepository.createQueryBuilder('following') | ||||
| 				.select('COUNT(DISTINCT following.followerHost)') | ||||
| 				.where('following.followerHost IS NOT NULL') | ||||
| 				.andWhere(meta.blockedHosts.length === 0 ? '1=1' : 'following.followerHost NOT IN (:...blocked)', { blocked: meta.blockedHosts }) | ||||
| 				.andWhere(meta.blockedHosts.length === 0 ? '1=1' : 'following.followerHost NOT ILIKE ANY(ARRAY[:...blocked])', { blocked: meta.blockedHosts.flatMap(x => [x, `%.${x}`]) }) | ||||
| 				.andWhere(`following.followerHost NOT IN (${ suspendedInstancesQuery.getQuery() })`) | ||||
| 				.getRawOne() | ||||
| 				.then(x => parseInt(x.count, 10)), | ||||
| 			this.followingsRepository.createQueryBuilder('following') | ||||
| 				.select('COUNT(DISTINCT following.followeeHost)') | ||||
| 				.where('following.followeeHost IS NOT NULL') | ||||
| 				.andWhere(meta.blockedHosts.length === 0 ? '1=1' : 'following.followeeHost NOT IN (:...blocked)', { blocked: meta.blockedHosts }) | ||||
| 				.andWhere(meta.blockedHosts.length === 0 ? '1=1' : 'following.followeeHost NOT ILIKE ANY(ARRAY[:...blocked])', { blocked: meta.blockedHosts.flatMap(x => [x, `%.${x}`]) }) | ||||
| 				.andWhere(`following.followeeHost NOT IN (${ suspendedInstancesQuery.getQuery() })`) | ||||
| 				.andWhere(`following.followeeHost IN (${ pubsubSubQuery.getQuery() })`) | ||||
| 				.setParameters(pubsubSubQuery.getParameters()) | ||||
| @@ -84,7 +84,7 @@ export default class FederationChart extends Chart<typeof schema> { | ||||
| 			this.instancesRepository.createQueryBuilder('instance') | ||||
| 				.select('COUNT(instance.id)') | ||||
| 				.where(`instance.host IN (${ subInstancesQuery.getQuery() })`) | ||||
| 				.andWhere(meta.blockedHosts.length === 0 ? '1=1' : 'instance.host NOT IN (:...blocked)', { blocked: meta.blockedHosts }) | ||||
| 				.andWhere(meta.blockedHosts.length === 0 ? '1=1' : 'instance.host NOT ILIKE ANY(ARRAY[:...blocked])', { blocked: meta.blockedHosts.flatMap(x => [x, `%.${x}`]) }) | ||||
| 				.andWhere('instance.isSuspended = false') | ||||
| 				.andWhere('instance.isNotResponding = false') | ||||
| 				.getRawOne() | ||||
| @@ -92,7 +92,7 @@ export default class FederationChart extends Chart<typeof schema> { | ||||
| 			this.instancesRepository.createQueryBuilder('instance') | ||||
| 				.select('COUNT(instance.id)') | ||||
| 				.where(`instance.host IN (${ pubInstancesQuery.getQuery() })`) | ||||
| 				.andWhere(meta.blockedHosts.length === 0 ? '1=1' : 'instance.host NOT IN (:...blocked)', { blocked: meta.blockedHosts }) | ||||
| 				.andWhere(meta.blockedHosts.length === 0 ? '1=1' : 'instance.host NOT ILIKE ANY(ARRAY[:...blocked])', { blocked: meta.blockedHosts.flatMap(x => [x, `%.${x}`]) }) | ||||
| 				.andWhere('instance.isSuspended = false') | ||||
| 				.andWhere('instance.isNotResponding = false') | ||||
| 				.getRawOne() | ||||
|   | ||||
| @@ -7,8 +7,8 @@ import type { } from '@/models/entities/Blocking.js'; | ||||
| import type { User } from '@/models/entities/User.js'; | ||||
| import type { Instance } from '@/models/entities/Instance.js'; | ||||
| import { MetaService } from '@/core/MetaService.js'; | ||||
| import { UtilityService } from '../UtilityService.js'; | ||||
| import { bindThis } from '@/decorators.js'; | ||||
| import { UserEntityService } from './UserEntityService.js'; | ||||
|  | ||||
| @Injectable() | ||||
| export class InstanceEntityService { | ||||
| @@ -17,6 +17,8 @@ export class InstanceEntityService { | ||||
| 		private instancesRepository: InstancesRepository, | ||||
|  | ||||
| 		private metaService: MetaService, | ||||
|  | ||||
| 		private utilityService: UtilityService, | ||||
| 	) { | ||||
| 	} | ||||
|  | ||||
| @@ -27,7 +29,7 @@ export class InstanceEntityService { | ||||
| 		const meta = await this.metaService.fetch(); | ||||
| 		return { | ||||
| 			id: instance.id, | ||||
| 			caughtAt: instance.caughtAt.toISOString(), | ||||
| 			firstRetrievedAt: instance.firstRetrievedAt.toISOString(), | ||||
| 			host: instance.host, | ||||
| 			usersCount: instance.usersCount, | ||||
| 			notesCount: instance.notesCount, | ||||
| @@ -35,7 +37,7 @@ export class InstanceEntityService { | ||||
| 			followersCount: instance.followersCount, | ||||
| 			isNotResponding: instance.isNotResponding, | ||||
| 			isSuspended: instance.isSuspended, | ||||
| 			isBlocked: meta.blockedHosts.includes(instance.host), | ||||
| 			isBlocked: this.utilityService.isBlockedHost(meta.blockedHosts, instance.host), | ||||
| 			softwareName: instance.softwareName, | ||||
| 			softwareVersion: instance.softwareVersion, | ||||
| 			openRegistrations: instance.openRegistrations, | ||||
|   | ||||
| @@ -114,6 +114,9 @@ export class NotificationEntityService implements OnModuleInit { | ||||
| 			...(notification.type === 'groupInvited' ? { | ||||
| 				invitation: this.userGroupInvitationEntityService.pack(notification.userGroupInvitationId!), | ||||
| 			} : {}), | ||||
| 			...(notification.type === 'achievementEarned' ? { | ||||
| 				achievement: notification.achievement, | ||||
| 			} : {}), | ||||
| 			...(notification.type === 'app' ? { | ||||
| 				body: notification.customBody, | ||||
| 				header: notification.customHeader ?? token?.name, | ||||
|   | ||||
							
								
								
									
										84
									
								
								packages/backend/src/core/entities/RoleEntityService.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										84
									
								
								packages/backend/src/core/entities/RoleEntityService.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,84 @@ | ||||
| import { Inject, Injectable } from '@nestjs/common'; | ||||
| import { DI } from '@/di-symbols.js'; | ||||
| import type { RoleAssignmentsRepository, RolesRepository } from '@/models/index.js'; | ||||
| import { awaitAll } from '@/misc/prelude/await-all.js'; | ||||
| import type { Packed } from '@/misc/schema.js'; | ||||
| import type { User } from '@/models/entities/User.js'; | ||||
| import type { Role } from '@/models/entities/Role.js'; | ||||
| import { bindThis } from '@/decorators.js'; | ||||
| import { DEFAULT_POLICIES } from '@/core/RoleService.js'; | ||||
| import { UserEntityService } from './UserEntityService.js'; | ||||
|  | ||||
| @Injectable() | ||||
| export class RoleEntityService { | ||||
| 	constructor( | ||||
| 		@Inject(DI.rolesRepository) | ||||
| 		private rolesRepository: RolesRepository, | ||||
|  | ||||
| 		@Inject(DI.roleAssignmentsRepository) | ||||
| 		private roleAssignmentsRepository: RoleAssignmentsRepository, | ||||
|  | ||||
| 		private userEntityService: UserEntityService, | ||||
| 	) { | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public async pack( | ||||
| 		src: Role['id'] | Role, | ||||
| 		me?: { id: User['id'] } | null | undefined, | ||||
| 		options?: { | ||||
| 			detail?: boolean; | ||||
| 		}, | ||||
| 	) { | ||||
| 		const opts = Object.assign({ | ||||
| 			detail: true, | ||||
| 		}, options); | ||||
|  | ||||
| 		const role = typeof src === 'object' ? src : await this.rolesRepository.findOneByOrFail({ id: src }); | ||||
|  | ||||
| 		const assigns = await this.roleAssignmentsRepository.findBy({ | ||||
| 			roleId: role.id, | ||||
| 		}); | ||||
|  | ||||
| 		const policies = { ...role.policies }; | ||||
| 		for (const [k, v] of Object.entries(DEFAULT_POLICIES)) { | ||||
| 			if (policies[k] == null) policies[k] = { | ||||
| 				useDefault: true, | ||||
| 				priority: 0, | ||||
| 				value: v, | ||||
| 			}; | ||||
| 		} | ||||
|  | ||||
| 		return await awaitAll({ | ||||
| 			id: role.id, | ||||
| 			createdAt: role.createdAt.toISOString(), | ||||
| 			updatedAt: role.updatedAt.toISOString(), | ||||
| 			name: role.name, | ||||
| 			description: role.description, | ||||
| 			color: role.color, | ||||
| 			target: role.target, | ||||
| 			condFormula: role.condFormula, | ||||
| 			isPublic: role.isPublic, | ||||
| 			isAdministrator: role.isAdministrator, | ||||
| 			isModerator: role.isModerator, | ||||
| 			canEditMembersByModerator: role.canEditMembersByModerator, | ||||
| 			policies: policies, | ||||
| 			usersCount: assigns.length, | ||||
| 			...(opts.detail ? { | ||||
| 				users: this.userEntityService.packMany(assigns.map(x => x.userId), me), | ||||
| 			} : {}), | ||||
| 		}); | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public packMany( | ||||
| 		roles: any[], | ||||
| 		me: { id: User['id'] }, | ||||
| 		options?: { | ||||
| 			detail?: boolean; | ||||
| 		}, | ||||
| 	) { | ||||
| 		return Promise.all(roles.map(x => this.pack(x, me, options))); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -12,7 +12,9 @@ import { Cache } from '@/misc/cache.js'; | ||||
| import type { Instance } from '@/models/entities/Instance.js'; | ||||
| import type { ILocalUser, IRemoteUser, User } from '@/models/entities/User.js'; | ||||
| import { birthdaySchema, descriptionSchema, localUsernameSchema, locationSchema, nameSchema, passwordSchema } from '@/models/entities/User.js'; | ||||
| import type { UsersRepository, UserSecurityKeysRepository, FollowingsRepository, FollowRequestsRepository, BlockingsRepository, MutingsRepository, DriveFilesRepository, NoteUnreadsRepository, ChannelFollowingsRepository, NotificationsRepository, UserNotePiningsRepository, UserProfilesRepository, InstancesRepository, AnnouncementReadsRepository, MessagingMessagesRepository, UserGroupJoiningsRepository, AnnouncementsRepository, AntennaNotesRepository, PagesRepository } from '@/models/index.js'; | ||||
| import type { UsersRepository, UserSecurityKeysRepository, FollowingsRepository, FollowRequestsRepository, BlockingsRepository, MutingsRepository, DriveFilesRepository, NoteUnreadsRepository, ChannelFollowingsRepository, NotificationsRepository, UserNotePiningsRepository, UserProfilesRepository, InstancesRepository, AnnouncementReadsRepository, MessagingMessagesRepository, UserGroupJoiningsRepository, AnnouncementsRepository, AntennaNotesRepository, PagesRepository, UserProfile } from '@/models/index.js'; | ||||
| import { bindThis } from '@/decorators.js'; | ||||
| import { RoleService } from '@/core/RoleService.js'; | ||||
| import type { OnModuleInit } from '@nestjs/common'; | ||||
| import type { AntennaService } from '../AntennaService.js'; | ||||
| import type { CustomEmojiService } from '../CustomEmojiService.js'; | ||||
| @@ -41,7 +43,6 @@ function isRemoteUser<T extends { host: User['host'] }>(user: T): user is T & { | ||||
| function isRemoteUser(user: User | { host: User['host'] }): boolean { | ||||
| 	return !isLocalUser(user); | ||||
| } | ||||
| import { bindThis } from '@/decorators.js'; | ||||
|  | ||||
| @Injectable() | ||||
| export class UserEntityService implements OnModuleInit { | ||||
| @@ -50,6 +51,7 @@ export class UserEntityService implements OnModuleInit { | ||||
| 	private pageEntityService: PageEntityService; | ||||
| 	private customEmojiService: CustomEmojiService; | ||||
| 	private antennaService: AntennaService; | ||||
| 	private roleService: RoleService; | ||||
| 	private userInstanceCache: Cache<Instance | null>; | ||||
|  | ||||
| 	constructor( | ||||
| @@ -120,6 +122,7 @@ export class UserEntityService implements OnModuleInit { | ||||
| 		//private pageEntityService: PageEntityService, | ||||
| 		//private customEmojiService: CustomEmojiService, | ||||
| 		//private antennaService: AntennaService, | ||||
| 		//private roleService: RoleService, | ||||
| 	) { | ||||
| 		this.userInstanceCache = new Cache<Instance | null>(1000 * 60 * 60 * 3); | ||||
| 	} | ||||
| @@ -130,6 +133,7 @@ export class UserEntityService implements OnModuleInit { | ||||
| 		this.pageEntityService = this.moduleRef.get('PageEntityService'); | ||||
| 		this.customEmojiService = this.moduleRef.get('CustomEmojiService'); | ||||
| 		this.antennaService = this.moduleRef.get('AntennaService'); | ||||
| 		this.roleService = this.moduleRef.get('RoleService'); | ||||
| 	} | ||||
|  | ||||
| 	//#region Validators | ||||
| @@ -339,6 +343,7 @@ export class UserEntityService implements OnModuleInit { | ||||
| 		options?: { | ||||
| 			detail?: D, | ||||
| 			includeSecrets?: boolean, | ||||
| 			userProfile?: UserProfile, | ||||
| 		}, | ||||
| 	): Promise<IsMeAndIsUserDetailed<ExpectsMe, D>> { | ||||
| 		const opts = Object.assign({ | ||||
| @@ -371,7 +376,7 @@ export class UserEntityService implements OnModuleInit { | ||||
| 			.innerJoinAndSelect('pin.note', 'note') | ||||
| 			.orderBy('pin.id', 'DESC') | ||||
| 			.getMany() : []; | ||||
| 		const profile = opts.detail ? await this.userProfilesRepository.findOneByOrFail({ userId: user.id }) : null; | ||||
| 		const profile = opts.detail ? (opts.userProfile ?? await this.userProfilesRepository.findOneByOrFail({ userId: user.id })) : null; | ||||
|  | ||||
| 		const followingCount = profile == null ? null : | ||||
| 			(profile.ffVisibility === 'public') || isMe ? user.followingCount : | ||||
| @@ -383,6 +388,9 @@ export class UserEntityService implements OnModuleInit { | ||||
| 			(profile.ffVisibility === 'followers') && (relation && relation.isFollowing) ? user.followersCount : | ||||
| 			null; | ||||
|  | ||||
| 		const isModerator = isMe && opts.detail ? this.roleService.isModerator(user) : null; | ||||
| 		const isAdmin = isMe && opts.detail ? this.roleService.isAdministrator(user) : null; | ||||
|  | ||||
| 		const falsy = opts.detail ? false : undefined; | ||||
|  | ||||
| 		const packed = { | ||||
| @@ -392,8 +400,6 @@ export class UserEntityService implements OnModuleInit { | ||||
| 			host: user.host, | ||||
| 			avatarUrl: this.getAvatarUrlSync(user), | ||||
| 			avatarBlurhash: user.avatar?.blurhash ?? null, | ||||
| 			isAdmin: user.isAdmin ?? falsy, | ||||
| 			isModerator: user.isModerator ?? falsy, | ||||
| 			isBot: user.isBot ?? falsy, | ||||
| 			isCat: user.isCat ?? falsy, | ||||
| 			instance: user.host ? this.userInstanceCache.fetch(user.host, | ||||
| @@ -418,7 +424,7 @@ export class UserEntityService implements OnModuleInit { | ||||
| 				bannerUrl: user.banner ? this.driveFileEntityService.getPublicUrl(user.banner, false) : null, | ||||
| 				bannerBlurhash: user.banner?.blurhash ?? null, | ||||
| 				isLocked: user.isLocked, | ||||
| 				isSilenced: user.isSilenced ?? falsy, | ||||
| 				isSilenced: this.roleService.getUserPolicies(user.id).then(r => !r.canPublicNote), | ||||
| 				isSuspended: user.isSuspended ?? falsy, | ||||
| 				description: profile!.description, | ||||
| 				location: profile!.location, | ||||
| @@ -443,14 +449,21 @@ export class UserEntityService implements OnModuleInit { | ||||
| 						userId: user.id, | ||||
| 					}).then(result => result >= 1) | ||||
| 					: false, | ||||
| 				...(isMe || opts.includeSecrets ? { | ||||
| 					driveCapacityOverrideMb: user.driveCapacityOverrideMb, | ||||
| 				} : {}), | ||||
| 				roles: this.roleService.getUserRoles(user.id).then(roles => roles.filter(role => role.isPublic).map(role => ({ | ||||
| 					id: role.id, | ||||
| 					name: role.name, | ||||
| 					color: role.color, | ||||
| 					description: role.description, | ||||
| 					isModerator: role.isModerator, | ||||
| 					isAdministrator: role.isAdministrator, | ||||
| 				}))), | ||||
| 			} : {}), | ||||
|  | ||||
| 			...(opts.detail && isMe ? { | ||||
| 				avatarId: user.avatarId, | ||||
| 				bannerId: user.bannerId, | ||||
| 				isModerator: isModerator, | ||||
| 				isAdmin: isAdmin, | ||||
| 				injectFeaturedNote: profile!.injectFeaturedNote, | ||||
| 				receiveAnnouncementEmail: profile!.receiveAnnouncementEmail, | ||||
| 				alwaysMarkNsfw: profile!.alwaysMarkNsfw, | ||||
| @@ -481,9 +494,12 @@ export class UserEntityService implements OnModuleInit { | ||||
| 				mutingNotificationTypes: profile!.mutingNotificationTypes, | ||||
| 				emailNotificationTypes: profile!.emailNotificationTypes, | ||||
| 				showTimelineReplies: user.showTimelineReplies ?? falsy, | ||||
| 				achievements: profile!.achievements, | ||||
| 				loggedInDays: profile!.loggedInDates.length, | ||||
| 			} : {}), | ||||
|  | ||||
| 			...(opts.includeSecrets ? { | ||||
| 				policies: this.roleService.getUserPolicies(user.id), | ||||
| 				email: profile!.email, | ||||
| 				emailVerified: profile!.emailVerified, | ||||
| 				securityKeysList: profile!.twoFactorEnabled | ||||
|   | ||||
| @@ -69,6 +69,8 @@ export const DI = { | ||||
| 	adsRepository: Symbol('adsRepository'), | ||||
| 	passwordResetRequestsRepository: Symbol('passwordResetRequestsRepository'), | ||||
| 	retentionAggregationsRepository: Symbol('retentionAggregationsRepository'), | ||||
| 	rolesRepository: Symbol('rolesRepository'), | ||||
| 	roleAssignmentsRepository: Symbol('roleAssignmentsRepository'), | ||||
| 	flashsRepository: Symbol('flashsRepository'), | ||||
| 	flashLikesRepository: Symbol('flashLikesRepository'), | ||||
| 	//#endregion | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| import { Module } from '@nestjs/common'; | ||||
| import { DI } from '@/di-symbols.js'; | ||||
| import { User, Note, Announcement, AnnouncementRead, App, NoteFavorite, NoteThreadMuting, NoteReaction, NoteUnread, Notification, Poll, PollVote, UserProfile, UserKeypair, UserPending, AttestationChallenge, UserSecurityKey, UserPublickey, UserList, UserListJoining, UserGroup, UserGroupJoining, UserGroupInvitation, UserNotePining, UserIp, UsedUsername, Following, FollowRequest, Instance, Emoji, DriveFile, DriveFolder, Meta, Muting, Blocking, SwSubscription, Hashtag, AbuseUserReport, RegistrationTicket, AuthSession, AccessToken, Signin, MessagingMessage, Page, PageLike, GalleryPost, GalleryLike, ModerationLog, Clip, ClipNote, Antenna, AntennaNote, PromoNote, PromoRead, Relay, MutedNote, Channel, ChannelFollowing, ChannelNotePining, RegistryItem, Webhook, Ad, PasswordResetRequest, RetentionAggregation, FlashLike, Flash } from './index.js'; | ||||
| import { User, Note, Announcement, AnnouncementRead, App, NoteFavorite, NoteThreadMuting, NoteReaction, NoteUnread, Notification, Poll, PollVote, UserProfile, UserKeypair, UserPending, AttestationChallenge, UserSecurityKey, UserPublickey, UserList, UserListJoining, UserGroup, UserGroupJoining, UserGroupInvitation, UserNotePining, UserIp, UsedUsername, Following, FollowRequest, Instance, Emoji, DriveFile, DriveFolder, Meta, Muting, Blocking, SwSubscription, Hashtag, AbuseUserReport, RegistrationTicket, AuthSession, AccessToken, Signin, MessagingMessage, Page, PageLike, GalleryPost, GalleryLike, ModerationLog, Clip, ClipNote, Antenna, AntennaNote, PromoNote, PromoRead, Relay, MutedNote, Channel, ChannelFollowing, ChannelNotePining, RegistryItem, Webhook, Ad, PasswordResetRequest, RetentionAggregation, FlashLike, Flash, Role, RoleAssignment } from './index.js'; | ||||
| import type { DataSource } from 'typeorm'; | ||||
| import type { Provider } from '@nestjs/common'; | ||||
|  | ||||
| @@ -400,6 +400,18 @@ const $flashLikesRepository: Provider = { | ||||
| 	inject: [DI.db], | ||||
| }; | ||||
|  | ||||
| const $rolesRepository: Provider = { | ||||
| 	provide: DI.rolesRepository, | ||||
| 	useFactory: (db: DataSource) => db.getRepository(Role), | ||||
| 	inject: [DI.db], | ||||
| }; | ||||
|  | ||||
| const $roleAssignmentsRepository: Provider = { | ||||
| 	provide: DI.roleAssignmentsRepository, | ||||
| 	useFactory: (db: DataSource) => db.getRepository(RoleAssignment), | ||||
| 	inject: [DI.db], | ||||
| }; | ||||
|  | ||||
| @Module({ | ||||
| 	imports: [ | ||||
| 	], | ||||
| @@ -468,6 +480,8 @@ const $flashLikesRepository: Provider = { | ||||
| 		$adsRepository, | ||||
| 		$passwordResetRequestsRepository, | ||||
| 		$retentionAggregationsRepository, | ||||
| 		$rolesRepository, | ||||
| 		$roleAssignmentsRepository, | ||||
| 		$flashsRepository, | ||||
| 		$flashLikesRepository, | ||||
| 	], | ||||
| @@ -536,6 +550,8 @@ const $flashLikesRepository: Provider = { | ||||
| 		$adsRepository, | ||||
| 		$passwordResetRequestsRepository, | ||||
| 		$retentionAggregationsRepository, | ||||
| 		$rolesRepository, | ||||
| 		$roleAssignmentsRepository, | ||||
| 		$flashsRepository, | ||||
| 		$flashLikesRepository, | ||||
| 	], | ||||
|   | ||||
| @@ -44,7 +44,7 @@ export class Flash { | ||||
| 	public user: User | null; | ||||
|  | ||||
| 	@Column('varchar', { | ||||
| 		length: 16384, | ||||
| 		length: 32768, | ||||
| 	}) | ||||
| 	public script: string; | ||||
|  | ||||
|   | ||||
| @@ -13,7 +13,7 @@ export class Instance { | ||||
| 	@Column('timestamp with time zone', { | ||||
| 		comment: 'The caught date of the Instance.', | ||||
| 	}) | ||||
| 	public caughtAt: Date; | ||||
| 	public firstRetrievedAt: Date; | ||||
|  | ||||
| 	/** | ||||
| 	 * ホスト | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user