Compare commits
278 Commits
Author | SHA1 | Date | |
---|---|---|---|
![]() |
61ffe7417c | ||
![]() |
7651353f39 | ||
![]() |
3f5b81060f | ||
![]() |
63dc66769f | ||
![]() |
e0fc8cbf8f | ||
![]() |
f9d1bc340e | ||
![]() |
0b269e79fd | ||
![]() |
6159cfd138 | ||
![]() |
6a5bbd335b | ||
![]() |
39e269db8c | ||
![]() |
70fe23a3ce | ||
![]() |
a6a8a7fb85 | ||
![]() |
6641b13b4c | ||
![]() |
5136b05c9b | ||
![]() |
803c2144f4 | ||
![]() |
b69a079514 | ||
![]() |
2aa800cd55 | ||
![]() |
6e61a36d05 | ||
![]() |
f80bf1fb1c | ||
![]() |
d465e85239 | ||
![]() |
deed25a2ff | ||
![]() |
a486716520 | ||
![]() |
2361e11e98 | ||
![]() |
cd1f2adca7 | ||
![]() |
a558767b7a | ||
![]() |
399ce9b999 | ||
![]() |
a94a0b5b0b | ||
![]() |
76faec2115 | ||
![]() |
33c4e57994 | ||
![]() |
bc23496998 | ||
![]() |
d35ad95c18 | ||
![]() |
5facd11592 | ||
![]() |
e1e885d6b2 | ||
![]() |
5b6695114f | ||
![]() |
71dd7f89e9 | ||
![]() |
21331e53fe | ||
![]() |
7afee5977f | ||
![]() |
d195b0dec7 | ||
![]() |
8a95e850ad | ||
![]() |
a4d74d7d7e | ||
![]() |
256e0db36d | ||
![]() |
d593c1358a | ||
![]() |
1ff14d81c1 | ||
![]() |
4369d12eec | ||
![]() |
91cc033eb5 | ||
![]() |
57543e6b44 | ||
![]() |
a1b8cd15c4 | ||
![]() |
73f06e591a | ||
![]() |
6f7cfa82b5 | ||
![]() |
ff97a003d1 | ||
![]() |
53c92e3e23 | ||
![]() |
13d13bc2f6 | ||
![]() |
03744a25ed | ||
![]() |
eac3bf8bff | ||
![]() |
2e1fbb5b16 | ||
![]() |
98b3517d36 | ||
![]() |
dee662705e | ||
![]() |
0da0cc80b9 | ||
![]() |
650187deaf | ||
![]() |
2e565cac2c | ||
![]() |
ac7537278c | ||
![]() |
f9a2e98831 | ||
![]() |
54f789bd55 | ||
![]() |
5ac9d13516 | ||
![]() |
2be1a39d13 | ||
![]() |
f3c5edc852 | ||
![]() |
30704e6de8 | ||
![]() |
41932ac409 | ||
![]() |
9843c596d8 | ||
![]() |
baf65bfa69 | ||
![]() |
6501f80fc7 | ||
![]() |
b037f6566b | ||
![]() |
0ec8ebeba3 | ||
![]() |
af1c9251fc | ||
![]() |
4ad399c593 | ||
![]() |
55a9646f23 | ||
![]() |
46017f5725 | ||
![]() |
c20ce12f86 | ||
![]() |
1e28db2396 | ||
![]() |
5f3640c7fd | ||
![]() |
d65e5f6794 | ||
![]() |
e67d7bc0ea | ||
![]() |
1139632f95 | ||
![]() |
b51a8c3f82 | ||
![]() |
0d7256678e | ||
![]() |
eea33d07fd | ||
![]() |
f599337320 | ||
![]() |
7df019db0e | ||
![]() |
04f92bd688 | ||
![]() |
505ecf6c1f | ||
![]() |
c9ec08704e | ||
![]() |
6a3039f7b7 | ||
![]() |
868c8fffb3 | ||
![]() |
faed3b438e | ||
![]() |
6c982629ea | ||
![]() |
110bbbc7dc | ||
![]() |
4ad0345f20 | ||
![]() |
9d84214462 | ||
![]() |
3f199c7113 | ||
![]() |
e9417fb741 | ||
![]() |
ee74df6823 | ||
![]() |
26630bae81 | ||
![]() |
9bde9edcf6 | ||
![]() |
a12f07c42b | ||
![]() |
e7334c4fb0 | ||
![]() |
38f9d1e764 | ||
![]() |
2dfed75402 | ||
![]() |
0c12e80106 | ||
![]() |
b7522f69e7 | ||
![]() |
24705a7e39 | ||
![]() |
8add8025a0 | ||
![]() |
32fa79d928 | ||
![]() |
534be6ff25 | ||
![]() |
f684c07567 | ||
![]() |
788ae2f6ca | ||
![]() |
572000f868 | ||
![]() |
57f5df2d22 | ||
![]() |
b2a67ba5ca | ||
![]() |
d78e15cc1a | ||
![]() |
ceab34f5f3 | ||
![]() |
3a62625bbc | ||
![]() |
ad6844ac4a | ||
![]() |
a8c252a613 | ||
![]() |
1d39f785f1 | ||
![]() |
4b8b29b862 | ||
![]() |
0d148bd23b | ||
![]() |
ebedb81e3f | ||
![]() |
d195406fdc | ||
![]() |
5173ed37f9 | ||
![]() |
825551d64f | ||
![]() |
449761bada | ||
![]() |
5859df389f | ||
![]() |
562b02310f | ||
![]() |
65ed702d87 | ||
![]() |
c559a9843f | ||
![]() |
88c3957085 | ||
![]() |
01778e11dc | ||
![]() |
9d9e8a3c4e | ||
![]() |
ed3e035ad6 | ||
![]() |
07f885fea8 | ||
![]() |
2cc98226ca | ||
![]() |
8a6f73c5ff | ||
![]() |
00e3453ce1 | ||
![]() |
16646dd77a | ||
![]() |
1f39d1fe26 | ||
![]() |
e8f3c587c9 | ||
![]() |
4b43745e7c | ||
![]() |
9db2f60053 | ||
![]() |
4610d8dfe3 | ||
![]() |
fa296efdf6 | ||
![]() |
d9d98f84bf | ||
![]() |
7c3143b8e5 | ||
![]() |
387fcd5c5d | ||
![]() |
ebc6437977 | ||
![]() |
dbc23b5d20 | ||
![]() |
843f1aed4f | ||
![]() |
e42938cad6 | ||
![]() |
2a41f6c383 | ||
![]() |
671d21a2c1 | ||
![]() |
515692d7a6 | ||
![]() |
00d28826b9 | ||
![]() |
5b38f76254 | ||
![]() |
ca7dbd6010 | ||
![]() |
133644e5a9 | ||
![]() |
04d60426c7 | ||
![]() |
8282bbd07c | ||
![]() |
7190bd00c9 | ||
![]() |
44b9539818 | ||
![]() |
b2ed4c9508 | ||
![]() |
c7b5c8b19e | ||
![]() |
f4bee24ccf | ||
![]() |
e9cb18c5aa | ||
![]() |
d8f33bc0af | ||
![]() |
663999556f | ||
![]() |
c5a12ca2c7 | ||
![]() |
7af0e38dd3 | ||
![]() |
7d9d1ae7c2 | ||
![]() |
cef448f0f2 | ||
![]() |
67d64c9365 | ||
![]() |
269af9d6b9 | ||
![]() |
d37a734379 | ||
![]() |
7cb13cf839 | ||
![]() |
d7dda8f6e3 | ||
![]() |
6670c72f8b | ||
![]() |
b21064ffa4 | ||
![]() |
1959cb462b | ||
![]() |
1d6767ef0c | ||
![]() |
4735ae6451 | ||
![]() |
452bd6db25 | ||
![]() |
f5d6b84381 | ||
![]() |
34f5d81d1f | ||
![]() |
aa8adc07aa | ||
![]() |
d87bb807c3 | ||
![]() |
7646d6ed47 | ||
![]() |
41a6ed0de0 | ||
![]() |
ec8074cd49 | ||
![]() |
7131eb1827 | ||
![]() |
605b0f27e4 | ||
![]() |
80d2e157f6 | ||
![]() |
1e3447bccb | ||
![]() |
5ffa106cc1 | ||
![]() |
fc641c9b96 | ||
![]() |
5f49ac1b11 | ||
![]() |
9ffecf25dc | ||
![]() |
35fd523edf | ||
![]() |
6721d4216c | ||
![]() |
e3275e916b | ||
![]() |
3ba5541a66 | ||
![]() |
945c50db1f | ||
![]() |
30dce42e03 | ||
![]() |
d4fb201d05 | ||
![]() |
2a2e8d0cf6 | ||
![]() |
520ed8cb4d | ||
![]() |
8cab16c824 | ||
![]() |
ae63a1f494 | ||
![]() |
117ac53505 | ||
![]() |
2c379732d2 | ||
![]() |
9ca1197759 | ||
![]() |
8d3283e2a5 | ||
![]() |
6589e8a390 | ||
![]() |
b62894ff56 | ||
![]() |
da274cd458 | ||
![]() |
a2268a95be | ||
![]() |
9fd1b35d95 | ||
![]() |
869854eae7 | ||
![]() |
238f923b41 | ||
![]() |
a5df2b0293 | ||
![]() |
e6eae558d3 | ||
![]() |
083fa53d9c | ||
![]() |
7b73dd2d62 | ||
![]() |
7028b7331b | ||
![]() |
eefebab530 | ||
![]() |
683ddbef3e | ||
![]() |
bd23522c76 | ||
![]() |
c1dfbe2623 | ||
![]() |
ed9facbb33 | ||
![]() |
26fbb3a560 | ||
![]() |
93dd0638ad | ||
![]() |
0d44129ae3 | ||
![]() |
0cffe60abc | ||
![]() |
8a6750278e | ||
![]() |
d347f0a087 | ||
![]() |
226e0c4714 | ||
![]() |
0b2f945bb6 | ||
![]() |
2f6c45e118 | ||
![]() |
a5f54580a9 | ||
![]() |
70df8c77fa | ||
![]() |
2c52655b17 | ||
![]() |
6c4c071ae9 | ||
![]() |
b19dba80f4 | ||
![]() |
a8b19f4aa8 | ||
![]() |
09f4b9e546 | ||
![]() |
2e6d8c792b | ||
![]() |
e6338a555d | ||
![]() |
313a489ba0 | ||
![]() |
b906ff3fed | ||
![]() |
ede96eca28 | ||
![]() |
42f3d9188b | ||
![]() |
a35e0e9261 | ||
![]() |
80a400a67c | ||
![]() |
7a6534f30b | ||
![]() |
68a523ec6d | ||
![]() |
97d6c1ee86 | ||
![]() |
19c93151ce | ||
![]() |
039a2af3ab | ||
![]() |
945129c371 | ||
![]() |
da32be3ef3 | ||
![]() |
468ec36830 | ||
![]() |
492fb9a115 | ||
![]() |
bd8b624bae | ||
![]() |
9dacf11702 | ||
![]() |
26ae2dfc0f | ||
![]() |
890564e1da | ||
![]() |
002f98987d | ||
![]() |
43956f3ffb | ||
![]() |
f2a9194c79 | ||
![]() |
4cd70df7f4 | ||
![]() |
21e4c3dfe9 |
@@ -114,11 +114,6 @@ id: 'aid'
|
|||||||
# IP address family used for outgoing request (ipv4, ipv6 or dual)
|
# IP address family used for outgoing request (ipv4, ipv6 or dual)
|
||||||
#outgoingAddressFamily: ipv4
|
#outgoingAddressFamily: ipv4
|
||||||
|
|
||||||
# Syslog option
|
|
||||||
#syslog:
|
|
||||||
# host: localhost
|
|
||||||
# port: 514
|
|
||||||
|
|
||||||
# Proxy for HTTP/HTTPS
|
# Proxy for HTTP/HTTPS
|
||||||
#proxy: http://127.0.0.1:3128
|
#proxy: http://127.0.0.1:3128
|
||||||
|
|
||||||
|
@@ -114,11 +114,6 @@ id: 'aid'
|
|||||||
# IP address family used for outgoing request (ipv4, ipv6 or dual)
|
# IP address family used for outgoing request (ipv4, ipv6 or dual)
|
||||||
#outgoingAddressFamily: ipv4
|
#outgoingAddressFamily: ipv4
|
||||||
|
|
||||||
# Syslog option
|
|
||||||
#syslog:
|
|
||||||
# host: localhost
|
|
||||||
# port: 514
|
|
||||||
|
|
||||||
# Proxy for HTTP/HTTPS
|
# Proxy for HTTP/HTTPS
|
||||||
#proxy: http://127.0.0.1:3128
|
#proxy: http://127.0.0.1:3128
|
||||||
|
|
||||||
@@ -135,6 +130,7 @@ proxyBypassHosts:
|
|||||||
#proxySmtp: socks5://127.0.0.1:1080 # use SOCKS5
|
#proxySmtp: socks5://127.0.0.1:1080 # use SOCKS5
|
||||||
|
|
||||||
# Media Proxy
|
# Media Proxy
|
||||||
|
# Reference Implementation: https://github.com/misskey-dev/media-proxy
|
||||||
#mediaProxy: https://example.com/proxy
|
#mediaProxy: https://example.com/proxy
|
||||||
|
|
||||||
# Proxy remote files (default: false)
|
# Proxy remote files (default: false)
|
||||||
|
@@ -16,9 +16,15 @@ files/
|
|||||||
misskey-assets/
|
misskey-assets/
|
||||||
fluent-emojis/
|
fluent-emojis/
|
||||||
.pnp.*
|
.pnp.*
|
||||||
|
|
||||||
|
# .yarn関連
|
||||||
.yarn/*
|
.yarn/*
|
||||||
!.yarn/patches
|
!.yarn/patches
|
||||||
!.yarn/plugins
|
!.yarn/plugins
|
||||||
!.yarn/releases
|
!.yarn/releases
|
||||||
!.yarn/sdks
|
!.yarn/sdks
|
||||||
!.yarn/versions
|
!.yarn/versions
|
||||||
|
|
||||||
|
.idea/
|
||||||
|
packages/*/.vscode/
|
||||||
|
packages/backend/test/docker-compose.yml
|
||||||
|
3
.dockleignore
Normal file
3
.dockleignore
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
DKL-DI-0005
|
||||||
|
DKL-DI-0006
|
||||||
|
DKL-LI-0003
|
4
.github/workflows/docker-develop.yml
vendored
4
.github/workflows/docker-develop.yml
vendored
@@ -14,6 +14,8 @@ jobs:
|
|||||||
steps:
|
steps:
|
||||||
- name: Check out the repo
|
- name: Check out the repo
|
||||||
uses: actions/checkout@v3.3.0
|
uses: actions/checkout@v3.3.0
|
||||||
|
- name: Set up Docker Buildx
|
||||||
|
uses: docker/setup-buildx-action@v2.3.0
|
||||||
- name: Docker meta
|
- name: Docker meta
|
||||||
id: meta
|
id: meta
|
||||||
uses: docker/metadata-action@v4
|
uses: docker/metadata-action@v4
|
||||||
@@ -31,3 +33,5 @@ jobs:
|
|||||||
push: true
|
push: true
|
||||||
tags: misskey/misskey:develop
|
tags: misskey/misskey:develop
|
||||||
labels: develop
|
labels: develop
|
||||||
|
cache-from: type=gha
|
||||||
|
cache-to: type=gha,mode=max
|
||||||
|
30
.github/workflows/dockle.yml
vendored
Normal file
30
.github/workflows/dockle.yml
vendored
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
---
|
||||||
|
name: Dockle
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches:
|
||||||
|
- master
|
||||||
|
- develop
|
||||||
|
pull_request:
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
dockle:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
env:
|
||||||
|
DOCKER_CONTENT_TRUST: 1
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v3.2.0
|
||||||
|
- run: |
|
||||||
|
curl -L -o dockle.deb "https://github.com/goodwithtech/dockle/releases/download/v0.4.10/dockle_0.4.10_Linux-64bit.deb"
|
||||||
|
sudo dpkg -i dockle.deb
|
||||||
|
- run: |
|
||||||
|
cp .config/docker_example.env .config/docker.env
|
||||||
|
cp ./docker-compose.yml.example ./docker-compose.yml
|
||||||
|
- run: |
|
||||||
|
docker compose up -d web
|
||||||
|
docker tag "$(docker compose images web | awk 'OFS=":" {print $4}' | tail -n +2)" misskey-web:latest
|
||||||
|
- run: |
|
||||||
|
cmd="dockle --exit-code 1 misskey-web:latest ${image_name}"
|
||||||
|
echo "> ${cmd}"
|
||||||
|
eval "${cmd}"
|
6
.github/workflows/test.yml
vendored
6
.github/workflows/test.yml
vendored
@@ -109,8 +109,12 @@ jobs:
|
|||||||
# https://github.com/cypress-io/cypress/issues/4351#issuecomment-559489091
|
# https://github.com/cypress-io/cypress/issues/4351#issuecomment-559489091
|
||||||
- name: ALSA Env
|
- 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
|
run: echo -e 'pcm.!default {\n type hw\n card 0\n}\n\nctl.!default {\n type hw\n card 0\n}' > ~/.asoundrc
|
||||||
|
# XXX: This tries reinstalling Cypress if the binary is not cached
|
||||||
|
# Remove this when the cache issue is fixed
|
||||||
|
- name: Cypress install
|
||||||
|
run: pnpm exec cypress install
|
||||||
- name: Cypress run
|
- name: Cypress run
|
||||||
uses: cypress-io/github-action@v4
|
uses: cypress-io/github-action@v5
|
||||||
with:
|
with:
|
||||||
install: false
|
install: false
|
||||||
start: pnpm start:test
|
start: pnpm start:test
|
||||||
|
1
.gitignore
vendored
1
.gitignore
vendored
@@ -32,6 +32,7 @@ coverage
|
|||||||
!/.config/example.yml
|
!/.config/example.yml
|
||||||
!/.config/docker_example.yml
|
!/.config/docker_example.yml
|
||||||
!/.config/docker_example.env
|
!/.config/docker_example.env
|
||||||
|
docker-compose.yml
|
||||||
|
|
||||||
# misskey
|
# misskey
|
||||||
/build
|
/build
|
||||||
|
@@ -1 +1 @@
|
|||||||
v18.12.1
|
v18.13.0
|
||||||
|
161
CHANGELOG.md
161
CHANGELOG.md
@@ -8,6 +8,166 @@
|
|||||||
|
|
||||||
You should also include the user name that made the change.
|
You should also include the user name that made the change.
|
||||||
-->
|
-->
|
||||||
|
## 13.5.4 (2023/02/09)
|
||||||
|
|
||||||
|
### Improvements
|
||||||
|
- Server: UIのHTML(ノートなどの特別なページを除く)のキャッシュ時間を15秒から30秒に
|
||||||
|
- i/notificationsのレートリミットを緩和
|
||||||
|
|
||||||
|
### Bugfixes
|
||||||
|
- fix(client): validate url to improve security
|
||||||
|
- fix(client): dateの初期値が正常に入らない時がある
|
||||||
|
|
||||||
|
## 13.5.3 (2023/02/09)
|
||||||
|
|
||||||
|
### Improvements
|
||||||
|
- Client: デッキにチャンネルカラムを追加
|
||||||
|
|
||||||
|
## 13.5.2 (2023/02/08)
|
||||||
|
|
||||||
|
### Changes
|
||||||
|
- Revert: perf(client): do not render custom emojis in user names
|
||||||
|
|
||||||
|
### Bugfixes
|
||||||
|
- Client: register_note_view_interruptor not working
|
||||||
|
- Client: ログイントークンの再生成が出来ない
|
||||||
|
|
||||||
|
## 13.5.0 (2023/02/08)
|
||||||
|
|
||||||
|
### Changes
|
||||||
|
- perf(client): do not render custom emojis in user names
|
||||||
|
|
||||||
|
### Improvements
|
||||||
|
- Client: disableShowingAnimatedImagesのデフォルト値をprefers-reduced-motionにする
|
||||||
|
- enhance(client): tweak medialist style
|
||||||
|
|
||||||
|
### Bugfixes
|
||||||
|
- fix docker health check
|
||||||
|
- Client: MkEmojiPickerでもChromeで検索ダイアログで変換確定するとそのまま検索されてしまうのを修正
|
||||||
|
- fix(mfm): default degree not used in rotate
|
||||||
|
- fix(server): validate urls from ap to improve security
|
||||||
|
|
||||||
|
## 13.4.0 (2023/02/05)
|
||||||
|
|
||||||
|
### Improvements
|
||||||
|
- ロールにアイコンを設定してユーザー名の横に表示できるように
|
||||||
|
- feat: timeline page for non-login users
|
||||||
|
- 実績の単なるラッキーの獲得確立を調整
|
||||||
|
- Add Thai language support
|
||||||
|
|
||||||
|
### Bugfixes
|
||||||
|
- fix(server): 自分のノートをお気に入りに登録しても実績解除される問題を修正
|
||||||
|
- fix(server): clean up file in FileServer
|
||||||
|
- fix(server): Deny UNIX domain socket
|
||||||
|
- fix(server): validate filename and emoji name to improve security
|
||||||
|
- fix(client): validate input response in aiscript
|
||||||
|
- fix(client): add webhook delete button
|
||||||
|
- fix(client): tweak notification style
|
||||||
|
- fix(client): インラインコードを折り返して表示する
|
||||||
|
|
||||||
|
## 13.3.3 (2023/02/04)
|
||||||
|
|
||||||
|
### Bugfixes
|
||||||
|
- Server: improve security
|
||||||
|
|
||||||
|
## 13.3.2 (2023/02/04)
|
||||||
|
|
||||||
|
### Improvements
|
||||||
|
- 外部メディアプロキシへの対応を強化しました
|
||||||
|
外部メディアプロキシのFastify実装を作りました
|
||||||
|
https://github.com/misskey-dev/media-proxy
|
||||||
|
- Server: improve performance
|
||||||
|
|
||||||
|
### Bugfixes
|
||||||
|
- Client: validate urls to improve security
|
||||||
|
|
||||||
|
## 13.3.1 (2023/02/04)
|
||||||
|
|
||||||
|
### Bugfixes
|
||||||
|
- Client: カスタム絵文字にアニメーション画像を再生しない設定が適用されていない問題を修正
|
||||||
|
- Client: オートコンプリートでUnicode絵文字がカスタム絵文字として表示されてしまうのを修正
|
||||||
|
- Client: Fix Vue-plyr CORS issue
|
||||||
|
- Client: validate urls to improve security
|
||||||
|
|
||||||
|
## 13.3.0 (2023/02/03)
|
||||||
|
### Changes
|
||||||
|
- twitter/github/discord連携機能が削除されました
|
||||||
|
- ハッシュタグごとのチャートが削除されました
|
||||||
|
- syslogのサポートが削除されました
|
||||||
|
|
||||||
|
### Improvements
|
||||||
|
- ロールで広告の非表示が有効になっている場合は最初から広告を非表示にするように
|
||||||
|
|
||||||
|
## 13.2.6 (2023/02/01)
|
||||||
|
### Changes
|
||||||
|
- docker-compose.ymlをdocker-compose.yml.exampleにしました。docker-compose.ymlとしてコピーしてから使用してください。
|
||||||
|
|
||||||
|
### Improvements
|
||||||
|
- 絵文字ピッカーのパフォーマンスを改善
|
||||||
|
- AiScriptを0.12.4に更新
|
||||||
|
|
||||||
|
### Bugfixes
|
||||||
|
- Server: リレーと通信できない問題を修正
|
||||||
|
- Client: classicモード使用時にwindowサイズによってdefaultに変更された後に、windowサイズが元に戻ったらclassicに戻すように修正 #9669
|
||||||
|
- Client: Chromeで検索ダイアログで変換確定するとそのまま検索されてしまう問題を修正
|
||||||
|
|
||||||
|
## 13.2.4 (2023/01/27)
|
||||||
|
### Improvements
|
||||||
|
- リモートカスタム絵文字表示時のパフォーマンスを改善
|
||||||
|
- Default to `animation: false` when prefers-reduced-motion is set
|
||||||
|
- リアクション履歴が公開なら、ログインしていなくても表示できるように
|
||||||
|
- tweak blur setting
|
||||||
|
- tweak custom emoji cache
|
||||||
|
|
||||||
|
### Bugfixes
|
||||||
|
- fix aggregation of retention
|
||||||
|
- ダッシュボードでオンラインユーザー数が表示されない問題を修正
|
||||||
|
- フォロー申請・フォローのボタンが、通知から消えている問題を修正
|
||||||
|
|
||||||
|
## 13.2.3 (2023/01/26)
|
||||||
|
### Improvements
|
||||||
|
- カスタム絵文字の更新をリアルタイムで反映するように
|
||||||
|
|
||||||
|
### Bugfixes
|
||||||
|
- turnstile-failed: missing-input-secret
|
||||||
|
|
||||||
|
## 13.2.2 (2023/01/25)
|
||||||
|
### Improvements
|
||||||
|
- サーバーのパフォーマンスを改善
|
||||||
|
|
||||||
|
### Bugfixes
|
||||||
|
- サインイン時に誤ったレートリミットがかかることがある問題を修正
|
||||||
|
- MFMのposition、rotate、scaleで小数が使えない問題を修正
|
||||||
|
|
||||||
|
## 13.2.1 (2023/01/24)
|
||||||
|
### Improvements
|
||||||
|
- デザインの調整
|
||||||
|
- サーバーのパフォーマンスを改善
|
||||||
|
|
||||||
|
## 13.2.0 (2023/01/23)
|
||||||
|
|
||||||
|
### Improvements
|
||||||
|
- onlyServer / onlyQueue オプションを復活
|
||||||
|
- 他人の実績閲覧時は獲得条件を表示しないように
|
||||||
|
- アニメーション減らすオプション有効時はリアクションのアニメーションを無効に
|
||||||
|
- カスタム絵文字一覧のパフォーマンスを改善
|
||||||
|
|
||||||
|
### Bugfixes
|
||||||
|
- Aiscript: button is not defined
|
||||||
|
|
||||||
|
## 13.1.7 (2023/01/22)
|
||||||
|
|
||||||
|
### Improvements
|
||||||
|
- 新たな実績を追加
|
||||||
|
- MFMにscaleタグを追加
|
||||||
|
|
||||||
|
## 13.1.4 (2023/01/22)
|
||||||
|
|
||||||
|
### Improvements
|
||||||
|
- 新たな実績を追加
|
||||||
|
|
||||||
|
### Bugfixes
|
||||||
|
- Client: ローカリゼーション更新時にリロードが繰り返されることがあるのを修正
|
||||||
|
|
||||||
## 13.1.3 (2023/01/22)
|
## 13.1.3 (2023/01/22)
|
||||||
|
|
||||||
@@ -65,6 +225,7 @@ You should also include the user name that made the change.
|
|||||||
- Node.js 18.x or later is required
|
- Node.js 18.x or later is required
|
||||||
- PostgreSQL 15.x is required
|
- PostgreSQL 15.x is required
|
||||||
- Misskey not using 15 specific features at 13.0.0, but may do so in the future.
|
- Misskey not using 15 specific features at 13.0.0, but may do so in the future.
|
||||||
|
- Docker環境でPostgreSQLのアップデートを行う際のガイドはこちら: https://github.com/misskey-dev/misskey/pull/9641#issue-1536336620
|
||||||
- Elasticsearchのサポートが削除されました
|
- Elasticsearchのサポートが削除されました
|
||||||
- 代わりに今後任意の検索プロバイダを設定できる仕組みを構想しています。その仕組みを使えば今まで通りElasticsearchも利用できます
|
- 代わりに今後任意の検索プロバイダを設定できる仕組みを構想しています。その仕組みを使えば今まで通りElasticsearchも利用できます
|
||||||
- Yarnからpnpmに移行されました
|
- Yarnからpnpmに移行されました
|
||||||
|
@@ -44,7 +44,7 @@ Thank you for your PR! Before creating a PR, please check the following:
|
|||||||
- Check if there are any documents that need to be created or updated due to this change.
|
- Check if there are any documents that need to be created or updated due to this change.
|
||||||
- If you have added a feature or fixed a bug, please add a test case if possible.
|
- If you have added a feature or fixed a bug, please add a test case if possible.
|
||||||
- Please make sure that tests and Lint are passed in advance.
|
- Please make sure that tests and Lint are passed in advance.
|
||||||
- You can run it with `yarn test` and `yarn lint`. [See more info](#testing)
|
- You can run it with `pnpm test` and `pnpm lint`. [See more info](#testing)
|
||||||
- If this PR includes UI changes, please attach a screenshot in the text.
|
- If this PR includes UI changes, please attach a screenshot in the text.
|
||||||
|
|
||||||
Thanks for your cooperation 🤗
|
Thanks for your cooperation 🤗
|
||||||
@@ -102,7 +102,7 @@ If your language is not listed in Crowdin, please open an issue.
|
|||||||
During development, it is useful to use the
|
During development, it is useful to use the
|
||||||
|
|
||||||
```
|
```
|
||||||
yarn dev
|
pnpm dev
|
||||||
```
|
```
|
||||||
|
|
||||||
command.
|
command.
|
||||||
@@ -112,7 +112,7 @@ command.
|
|||||||
- Service Worker is watched by esbuild.
|
- Service Worker is watched by esbuild.
|
||||||
|
|
||||||
## Testing
|
## Testing
|
||||||
- Test codes are located in [`/test`](/test).
|
- Test codes are located in [`/packages/backend/test`](/packages/backend/test).
|
||||||
|
|
||||||
### Run test
|
### Run test
|
||||||
Create a config file.
|
Create a config file.
|
||||||
@@ -121,18 +121,18 @@ cp .github/misskey/test.yml .config/
|
|||||||
```
|
```
|
||||||
Prepare DB/Redis for testing.
|
Prepare DB/Redis for testing.
|
||||||
```
|
```
|
||||||
docker-compose -f packages/backend/test/docker-compose.yml up
|
docker compose -f packages/backend/test/docker-compose.yml up
|
||||||
```
|
```
|
||||||
Alternatively, prepare an empty (data can be erased) DB and edit `.config/test.yml`.
|
Alternatively, prepare an empty (data can be erased) DB and edit `.config/test.yml`.
|
||||||
|
|
||||||
Run all test.
|
Run all test.
|
||||||
```
|
```
|
||||||
yarn test
|
pnpm test
|
||||||
```
|
```
|
||||||
|
|
||||||
#### Run specify test
|
#### Run specify test
|
||||||
```
|
```
|
||||||
yarn jest -- foo.ts
|
pnpm jest -- foo.ts
|
||||||
```
|
```
|
||||||
|
|
||||||
### e2e tests
|
### e2e tests
|
||||||
@@ -177,9 +177,9 @@ vue-routerとの最大の違いは、niraxは複数のルーターが存在す
|
|||||||
これにより、アプリ内ウィンドウでブラウザとは個別にルーティングすることなどが可能になります。
|
これにより、アプリ内ウィンドウでブラウザとは個別にルーティングすることなどが可能になります。
|
||||||
|
|
||||||
## Notes
|
## Notes
|
||||||
### How to resolve conflictions occurred at yarn.lock?
|
### How to resolve conflictions occurred at pnpm-lock.yaml?
|
||||||
|
|
||||||
Just execute `yarn` to fix it.
|
Just execute `pnpm` to fix it.
|
||||||
|
|
||||||
### INSERTするときにはsaveではなくinsertを使用する
|
### INSERTするときにはsaveではなくinsertを使用する
|
||||||
#6441
|
#6441
|
||||||
@@ -265,7 +265,7 @@ MongoDBは`null`で返してきてたので、その感覚で`if (x === null)`
|
|||||||
### Migration作成方法
|
### Migration作成方法
|
||||||
packages/backendで:
|
packages/backendで:
|
||||||
```sh
|
```sh
|
||||||
yarn dlx typeorm migration:generate -d ormconfig.js -o <migration name>
|
pnpm dlx typeorm migration:generate -d ormconfig.js -o <migration name>
|
||||||
```
|
```
|
||||||
|
|
||||||
- 生成後、ファイルをmigration下に移してください
|
- 生成後、ファイルをmigration下に移してください
|
||||||
|
27
Dockerfile
27
Dockerfile
@@ -2,8 +2,12 @@ ARG NODE_VERSION=18.13.0-bullseye
|
|||||||
|
|
||||||
FROM node:${NODE_VERSION} AS builder
|
FROM node:${NODE_VERSION} AS builder
|
||||||
|
|
||||||
RUN apt-get update \
|
RUN --mount=type=cache,target=/var/cache/apt,sharing=locked \
|
||||||
&& apt-get install -y --no-install-recommends \
|
--mount=type=cache,target=/var/lib/apt,sharing=locked \
|
||||||
|
rm -f /etc/apt/apt.conf.d/docker-clean \
|
||||||
|
; echo 'Binary::apt::APT::Keep-Downloaded-Packages "true";' > /etc/apt/apt.conf.d/keep-cache \
|
||||||
|
&& apt-get update \
|
||||||
|
&& apt-get install -yqq --no-install-recommends \
|
||||||
build-essential
|
build-essential
|
||||||
|
|
||||||
RUN corepack enable
|
RUN corepack enable
|
||||||
@@ -16,7 +20,8 @@ COPY ["packages/backend/package.json", "./packages/backend/"]
|
|||||||
COPY ["packages/frontend/package.json", "./packages/frontend/"]
|
COPY ["packages/frontend/package.json", "./packages/frontend/"]
|
||||||
COPY ["packages/sw/package.json", "./packages/sw/"]
|
COPY ["packages/sw/package.json", "./packages/sw/"]
|
||||||
|
|
||||||
RUN pnpm i --frozen-lockfile
|
RUN --mount=type=cache,target=/root/.local/share/pnpm/store,sharing=locked \
|
||||||
|
pnpm i --frozen-lockfile --aggregate-output
|
||||||
|
|
||||||
COPY . ./
|
COPY . ./
|
||||||
|
|
||||||
@@ -24,20 +29,25 @@ ARG NODE_ENV=production
|
|||||||
|
|
||||||
RUN git submodule update --init
|
RUN git submodule update --init
|
||||||
RUN pnpm build
|
RUN pnpm build
|
||||||
|
RUN rm -rf .git/
|
||||||
|
|
||||||
FROM node:${NODE_VERSION}-slim AS runner
|
FROM node:${NODE_VERSION}-slim AS runner
|
||||||
|
|
||||||
ARG UID="991"
|
ARG UID="991"
|
||||||
ARG GID="991"
|
ARG GID="991"
|
||||||
|
|
||||||
RUN apt-get update \
|
RUN --mount=type=cache,target=/var/cache/apt,sharing=locked \
|
||||||
|
--mount=type=cache,target=/var/lib/apt,sharing=locked \
|
||||||
|
rm -f /etc/apt/apt.conf.d/docker-clean \
|
||||||
|
; echo 'Binary::apt::APT::Keep-Downloaded-Packages "true";' > /etc/apt/apt.conf.d/keep-cache \
|
||||||
|
&& apt-get update \
|
||||||
&& apt-get install -y --no-install-recommends \
|
&& apt-get install -y --no-install-recommends \
|
||||||
ffmpeg tini \
|
ffmpeg tini curl \
|
||||||
&& apt-get -y clean \
|
|
||||||
&& rm -rf /var/lib/apt/lists/* \
|
|
||||||
&& corepack enable \
|
&& corepack enable \
|
||||||
&& groupadd -g "${GID}" misskey \
|
&& groupadd -g "${GID}" misskey \
|
||||||
&& useradd -l -u "${UID}" -g "${GID}" -m -d /misskey misskey
|
&& useradd -l -u "${UID}" -g "${GID}" -m -d /misskey misskey \
|
||||||
|
&& find / -type f -perm /u+s -ignore_readdir_race -exec chmod u-s {} \; \
|
||||||
|
&& find / -type f -perm /g+s -ignore_readdir_race -exec chmod g-s {} \;
|
||||||
|
|
||||||
USER misskey
|
USER misskey
|
||||||
WORKDIR /misskey
|
WORKDIR /misskey
|
||||||
@@ -51,5 +61,6 @@ COPY --chown=misskey:misskey --from=builder /misskey/fluent-emojis /misskey/flue
|
|||||||
COPY --chown=misskey:misskey . ./
|
COPY --chown=misskey:misskey . ./
|
||||||
|
|
||||||
ENV NODE_ENV=production
|
ENV NODE_ENV=production
|
||||||
|
HEALTHCHECK --interval=5s --retries=20 CMD ["/bin/bash", "/misskey/healthcheck.sh"]
|
||||||
ENTRYPOINT ["/usr/bin/tini", "--"]
|
ENTRYPOINT ["/usr/bin/tini", "--"]
|
||||||
CMD ["pnpm", "run", "migrateandstart"]
|
CMD ["pnpm", "run", "migrateandstart"]
|
||||||
|
@@ -24,6 +24,8 @@
|
|||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
|
[](https://codecov.io/gh/misskey-dev/misskey)
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div>
|
<div>
|
||||||
|
@@ -6,16 +6,13 @@ Also, the later tasks are more indefinite and are subject to change as developme
|
|||||||
This is the phase we are at now. We need to make a high-maintenance environment that can withstand future development.
|
This is the phase we are at now. We need to make a high-maintenance environment that can withstand future development.
|
||||||
|
|
||||||
- Make the number of type errors zero (backend)
|
- Make the number of type errors zero (backend)
|
||||||
- Probably need to switch some libraries to others that make it difficult to reduce type errors
|
|
||||||
- e.g. koa to fastify https://github.com/misskey-dev/misskey/issues/7537
|
|
||||||
- Improve CI
|
- Improve CI
|
||||||
- Fix tests
|
- Fix tests
|
||||||
- mocha, jest, etc. do not support the combination of `TypeScript + ESM + Path alias`, and the tests currently do not work.
|
|
||||||
- Fix random test failures - https://github.com/misskey-dev/misskey/issues/7985 and https://github.com/misskey-dev/misskey/issues/7986
|
- Fix random test failures - https://github.com/misskey-dev/misskey/issues/7985 and https://github.com/misskey-dev/misskey/issues/7986
|
||||||
- Add more tests
|
- Add more tests
|
||||||
- May need to implement a mechanism that allows for DI
|
- ~~May need to implement a mechanism that allows for DI~~ → Done ✔️
|
||||||
- https://github.com/misskey-dev/misskey/pull/9085
|
- https://github.com/misskey-dev/misskey/pull/9085
|
||||||
- Measure coverage
|
- ~~Measure coverage~~ → Done ✔️
|
||||||
- https://github.com/misskey-dev/misskey/pull/9081
|
- https://github.com/misskey-dev/misskey/pull/9081
|
||||||
- Improve documentation
|
- Improve documentation
|
||||||
- Refactoring
|
- Refactoring
|
||||||
|
@@ -133,11 +133,6 @@ id: "aid"
|
|||||||
# IP address family used for outgoing request (ipv4, ipv6 or dual)
|
# IP address family used for outgoing request (ipv4, ipv6 or dual)
|
||||||
#outgoingAddressFamily: ipv4
|
#outgoingAddressFamily: ipv4
|
||||||
|
|
||||||
# Syslog option
|
|
||||||
#syslog:
|
|
||||||
# host: localhost
|
|
||||||
# port: 514
|
|
||||||
|
|
||||||
# Proxy for HTTP/HTTPS
|
# Proxy for HTTP/HTTPS
|
||||||
#proxy: http://127.0.0.1:3128
|
#proxy: http://127.0.0.1:3128
|
||||||
|
|
||||||
|
@@ -20,7 +20,7 @@ gulp.task('copy:frontend:fonts', () =>
|
|||||||
);
|
);
|
||||||
|
|
||||||
gulp.task('copy:frontend:tabler-icons', () =>
|
gulp.task('copy:frontend:tabler-icons', () =>
|
||||||
gulp.src('./packages/frontend/node_modules/@tabler/icons/iconfont/**/*').pipe(gulp.dest('./built/_frontend_dist_/tabler-icons/'))
|
gulp.src('./packages/frontend/node_modules/@tabler/icons-webfont/**/*').pipe(gulp.dest('./built/_frontend_dist_/tabler-icons/'))
|
||||||
);
|
);
|
||||||
|
|
||||||
gulp.task('copy:frontend:locales', cb => {
|
gulp.task('copy:frontend:locales', cb => {
|
||||||
|
4
healthcheck.sh
Normal file
4
healthcheck.sh
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
PORT=$(grep '^port:' /misskey/.config/default.yml | awk 'NR==1{print $2; exit}')
|
||||||
|
curl -s -S -o /dev/null "http://localhost:${PORT}"
|
@@ -1345,5 +1345,6 @@ _deck:
|
|||||||
tl: "الخيط الزمني"
|
tl: "الخيط الزمني"
|
||||||
antenna: "الهوائيات"
|
antenna: "الهوائيات"
|
||||||
list: "القوائم"
|
list: "القوائم"
|
||||||
|
channel: "القنوات"
|
||||||
mentions: "الإشارات"
|
mentions: "الإشارات"
|
||||||
direct: "مباشرة"
|
direct: "مباشرة"
|
||||||
|
@@ -1441,5 +1441,6 @@ _deck:
|
|||||||
tl: "টাইমলাইন"
|
tl: "টাইমলাইন"
|
||||||
antenna: "অ্যান্টেনা"
|
antenna: "অ্যান্টেনা"
|
||||||
list: "লিস্ট"
|
list: "লিস্ট"
|
||||||
|
channel: "চ্যানেলগুলি"
|
||||||
mentions: "উল্লেখসমূহ"
|
mentions: "উল্লেখসমূহ"
|
||||||
direct: "ডাইরেক্ট নোটগুলি"
|
direct: "ডাইরেক্ট নোটগুলি"
|
||||||
|
@@ -804,4 +804,5 @@ _deck:
|
|||||||
tl: "Časová osa"
|
tl: "Časová osa"
|
||||||
antenna: "Antény"
|
antenna: "Antény"
|
||||||
list: "Seznamy"
|
list: "Seznamy"
|
||||||
|
channel: "Kanály"
|
||||||
mentions: "Zmínění"
|
mentions: "Zmínění"
|
||||||
|
@@ -68,7 +68,7 @@ export: "Export"
|
|||||||
files: "Dateien"
|
files: "Dateien"
|
||||||
download: "Herunterladen"
|
download: "Herunterladen"
|
||||||
driveFileDeleteConfirm: "Möchtest du die Datei „{name}“ wirklich löschen? Notizen mit dieser Datei werden ebenso verschwinden."
|
driveFileDeleteConfirm: "Möchtest du die Datei „{name}“ wirklich löschen? Notizen mit dieser Datei werden ebenso verschwinden."
|
||||||
unfollowConfirm: "Möchtest du {name} nicht mehr folgen?"
|
unfollowConfirm: "Möchtest du {name} wirklich nicht mehr folgen?"
|
||||||
exportRequested: "Du hast einen Export angefragt. Dies kann etwas Zeit in Anspruch nehmen. Sobald der Export abgeschlossen ist, wird er deiner Drive hinzugefügt."
|
exportRequested: "Du hast einen Export angefragt. Dies kann etwas Zeit in Anspruch nehmen. Sobald der Export abgeschlossen ist, wird er deiner Drive hinzugefügt."
|
||||||
importRequested: "Du hast einen Import angefragt. Dies kann etwas Zeit in Anspruch nehmen."
|
importRequested: "Du hast einen Import angefragt. Dies kann etwas Zeit in Anspruch nehmen."
|
||||||
lists: "Listen"
|
lists: "Listen"
|
||||||
@@ -94,7 +94,7 @@ defaultNoteVisibility: "Standardsichtbarkeit"
|
|||||||
follow: "Folgen"
|
follow: "Folgen"
|
||||||
followRequest: "Follow-Anfrage senden"
|
followRequest: "Follow-Anfrage senden"
|
||||||
followRequests: "Follow-Anfragen"
|
followRequests: "Follow-Anfragen"
|
||||||
unfollow: "Nicht mehr folgen"
|
unfollow: "Entfolgen"
|
||||||
followRequestPending: "Follow-Anfrage ausstehend"
|
followRequestPending: "Follow-Anfrage ausstehend"
|
||||||
enterEmoji: "Gib ein Emoji ein"
|
enterEmoji: "Gib ein Emoji ein"
|
||||||
renote: "Renote"
|
renote: "Renote"
|
||||||
@@ -129,6 +129,7 @@ unblockConfirm: "Möchtest du diese Blockierung wirklich aufheben?"
|
|||||||
suspendConfirm: "Möchtest du diesen Benutzer wirklich sperren?"
|
suspendConfirm: "Möchtest du diesen Benutzer wirklich sperren?"
|
||||||
unsuspendConfirm: "Möchtest du diesen Benutzer wirklich entsperren?"
|
unsuspendConfirm: "Möchtest du diesen Benutzer wirklich entsperren?"
|
||||||
selectList: "Liste auswählen"
|
selectList: "Liste auswählen"
|
||||||
|
selectChannel: "Kanal auswählen"
|
||||||
selectAntenna: "Antenne auswählen"
|
selectAntenna: "Antenne auswählen"
|
||||||
selectWidget: "Widget auswählen"
|
selectWidget: "Widget auswählen"
|
||||||
editWidgets: "Widgets bearbeiten"
|
editWidgets: "Widgets bearbeiten"
|
||||||
@@ -1048,6 +1049,9 @@ _achievements:
|
|||||||
_noteFavorited1:
|
_noteFavorited1:
|
||||||
title: "Sternengucker"
|
title: "Sternengucker"
|
||||||
description: "Eine Notiz als Favorit markiert"
|
description: "Eine Notiz als Favorit markiert"
|
||||||
|
_myNoteFavorited1:
|
||||||
|
title: "Sternensucher"
|
||||||
|
description: "Ein anderer Benutzer hat eine deiner Notizen als Favoriten markiert"
|
||||||
_profileFilled:
|
_profileFilled:
|
||||||
title: "Perfekte Vorbereitung"
|
title: "Perfekte Vorbereitung"
|
||||||
description: "Fülle dein Profil aus"
|
description: "Fülle dein Profil aus"
|
||||||
@@ -1101,9 +1105,12 @@ _achievements:
|
|||||||
title: "I Love Misskey"
|
title: "I Love Misskey"
|
||||||
description: "Sende \"I ❤ #Misskey\""
|
description: "Sende \"I ❤ #Misskey\""
|
||||||
flavor: "Danke, dass du Misskey verwendest! - vom Entwicklerteam"
|
flavor: "Danke, dass du Misskey verwendest! - vom Entwicklerteam"
|
||||||
|
_foundTreasure:
|
||||||
|
title: "Schatzsuche"
|
||||||
|
description: "Du hast einen verborgenen Schatz gefunden"
|
||||||
_client30min:
|
_client30min:
|
||||||
title: "Kleine Pause"
|
title: "Kurze Pause"
|
||||||
description: "Seit dem Öffnen deines Clients sind 30 Minuten vergangen"
|
description: "Habe Misskey für 30 Minuten geöffnet"
|
||||||
_noteDeletedWithin1min:
|
_noteDeletedWithin1min:
|
||||||
title: "Ups"
|
title: "Ups"
|
||||||
description: "Lösche eine Notiz innerhalb von 1 Minute nachdem sie gesendet wurde"
|
description: "Lösche eine Notiz innerhalb von 1 Minute nachdem sie gesendet wurde"
|
||||||
@@ -1121,6 +1128,9 @@ _achievements:
|
|||||||
_htl20npm:
|
_htl20npm:
|
||||||
title: "Fließende Chronik"
|
title: "Fließende Chronik"
|
||||||
description: "Deine Startseitenchronik erreicht eine Geschwindigkeit von 20 npm (Notizen pro Minute)"
|
description: "Deine Startseitenchronik erreicht eine Geschwindigkeit von 20 npm (Notizen pro Minute)"
|
||||||
|
_viewInstanceChart:
|
||||||
|
title: "Analyst"
|
||||||
|
description: "Schau dir die Messwerte der Instanz an"
|
||||||
_outputHelloWorldOnScratchpad:
|
_outputHelloWorldOnScratchpad:
|
||||||
title: "Hallo Welt!"
|
title: "Hallo Welt!"
|
||||||
description: "Gib \"hello world\" in der Testumgebung aus"
|
description: "Gib \"hello world\" in der Testumgebung aus"
|
||||||
@@ -1186,6 +1196,9 @@ _role:
|
|||||||
baseRole: "Rollenvorlage"
|
baseRole: "Rollenvorlage"
|
||||||
useBaseValue: "Wert der Rollenvorlage verwenden"
|
useBaseValue: "Wert der Rollenvorlage verwenden"
|
||||||
chooseRoleToAssign: "Zuzuweisende Rolle auswählen"
|
chooseRoleToAssign: "Zuzuweisende Rolle auswählen"
|
||||||
|
iconUrl: "Icon-URL"
|
||||||
|
asBadge: "Als Abzeichen anzeigen"
|
||||||
|
descriptionOfAsBadge: "Ist dies aktiviert, so wird das Icon dieser Rolle an der Seite der Namen von Benutzern mit dieser Rolle angezeigt."
|
||||||
canEditMembersByModerator: "Moderatoren können Benutzern diese Rolle zuweisen"
|
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."
|
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: "Priorität"
|
||||||
@@ -1857,5 +1870,6 @@ _deck:
|
|||||||
tl: "Chronik"
|
tl: "Chronik"
|
||||||
antenna: "Antennen"
|
antenna: "Antennen"
|
||||||
list: "Listen"
|
list: "Listen"
|
||||||
|
channel: "Kanal"
|
||||||
mentions: "Erwähnungen"
|
mentions: "Erwähnungen"
|
||||||
direct: "Direktnachrichten"
|
direct: "Direktnachrichten"
|
||||||
|
@@ -68,7 +68,7 @@ export: "Export"
|
|||||||
files: "Files"
|
files: "Files"
|
||||||
download: "Download"
|
download: "Download"
|
||||||
driveFileDeleteConfirm: "Are you sure you want to delete the file \"{name}\"? Notes with this file attached will also be deleted."
|
driveFileDeleteConfirm: "Are you sure you want to delete the file \"{name}\"? Notes with this file attached will also be deleted."
|
||||||
unfollowConfirm: "Are you sure that you want to unfollow {name}?"
|
unfollowConfirm: "Are you sure you want to unfollow {name}?"
|
||||||
exportRequested: "You've requested an export. This may take a while. It will be added to your Drive once completed."
|
exportRequested: "You've requested an export. This may take a while. It will be added to your Drive once completed."
|
||||||
importRequested: "You've requested an import. This may take a while."
|
importRequested: "You've requested an import. This may take a while."
|
||||||
lists: "Lists"
|
lists: "Lists"
|
||||||
@@ -129,6 +129,7 @@ unblockConfirm: "Are you sure that you want to unblock this account?"
|
|||||||
suspendConfirm: "Are you sure that you want to suspend this account?"
|
suspendConfirm: "Are you sure that you want to suspend this account?"
|
||||||
unsuspendConfirm: "Are you sure that you want to unsuspend this account?"
|
unsuspendConfirm: "Are you sure that you want to unsuspend this account?"
|
||||||
selectList: "Select a list"
|
selectList: "Select a list"
|
||||||
|
selectChannel: "Select a channel"
|
||||||
selectAntenna: "Select an antenna"
|
selectAntenna: "Select an antenna"
|
||||||
selectWidget: "Select a widget"
|
selectWidget: "Select a widget"
|
||||||
editWidgets: "Edit widgets"
|
editWidgets: "Edit widgets"
|
||||||
@@ -945,7 +946,7 @@ _achievements:
|
|||||||
_notes1:
|
_notes1:
|
||||||
title: "just setting up my msky"
|
title: "just setting up my msky"
|
||||||
description: "Post your first note"
|
description: "Post your first note"
|
||||||
flavor: "Have a good Misskey life!"
|
flavor: "Have a good time with Misskey!"
|
||||||
_notes10:
|
_notes10:
|
||||||
title: "Some notes"
|
title: "Some notes"
|
||||||
description: "Post 10 notes"
|
description: "Post 10 notes"
|
||||||
@@ -1048,6 +1049,9 @@ _achievements:
|
|||||||
_noteFavorited1:
|
_noteFavorited1:
|
||||||
title: "Stargazer"
|
title: "Stargazer"
|
||||||
description: "Favorite your first note"
|
description: "Favorite your first note"
|
||||||
|
_myNoteFavorited1:
|
||||||
|
title: "Seeking Stars"
|
||||||
|
description: "Have somebody else favorite one of your notes"
|
||||||
_profileFilled:
|
_profileFilled:
|
||||||
title: "Well-prepared"
|
title: "Well-prepared"
|
||||||
description: "Set up your profile"
|
description: "Set up your profile"
|
||||||
@@ -1101,6 +1105,9 @@ _achievements:
|
|||||||
title: "I Love Misskey"
|
title: "I Love Misskey"
|
||||||
description: "Post \"I ❤ #Misskey\""
|
description: "Post \"I ❤ #Misskey\""
|
||||||
flavor: "Misskey's development team greatly appreciates your support!"
|
flavor: "Misskey's development team greatly appreciates your support!"
|
||||||
|
_foundTreasure:
|
||||||
|
title: "Treasure Hunt"
|
||||||
|
description: "You've found the hidden treasure"
|
||||||
_client30min:
|
_client30min:
|
||||||
title: "Short break"
|
title: "Short break"
|
||||||
description: "Spend 30 minutes on Misskey"
|
description: "Spend 30 minutes on Misskey"
|
||||||
@@ -1121,6 +1128,9 @@ _achievements:
|
|||||||
_htl20npm:
|
_htl20npm:
|
||||||
title: "Flowing Timeline"
|
title: "Flowing Timeline"
|
||||||
description: "Have the speed of your home timeline exceed 20 npm (notes per minute)"
|
description: "Have the speed of your home timeline exceed 20 npm (notes per minute)"
|
||||||
|
_viewInstanceChart:
|
||||||
|
title: "Analyst"
|
||||||
|
description: "View your instance's charts"
|
||||||
_outputHelloWorldOnScratchpad:
|
_outputHelloWorldOnScratchpad:
|
||||||
title: "Hello, world!"
|
title: "Hello, world!"
|
||||||
description: "Output \"hello world\" in the Scratchpad"
|
description: "Output \"hello world\" in the Scratchpad"
|
||||||
@@ -1186,6 +1196,9 @@ _role:
|
|||||||
baseRole: "Role template"
|
baseRole: "Role template"
|
||||||
useBaseValue: "Use role template value"
|
useBaseValue: "Use role template value"
|
||||||
chooseRoleToAssign: "Select the role to assign"
|
chooseRoleToAssign: "Select the role to assign"
|
||||||
|
iconUrl: "Icon URL"
|
||||||
|
asBadge: "Show as badge"
|
||||||
|
descriptionOfAsBadge: "This role's icon will be displayed next to the username of users with this role if turned on."
|
||||||
canEditMembersByModerator: "Allow moderators to edit the list of members for this role"
|
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."
|
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: "Priority"
|
||||||
@@ -1857,5 +1870,6 @@ _deck:
|
|||||||
tl: "Timeline"
|
tl: "Timeline"
|
||||||
antenna: "Antennas"
|
antenna: "Antennas"
|
||||||
list: "List"
|
list: "List"
|
||||||
|
channel: "Channel"
|
||||||
mentions: "Mentions"
|
mentions: "Mentions"
|
||||||
direct: "Direct notes"
|
direct: "Direct notes"
|
||||||
|
@@ -129,6 +129,7 @@ unblockConfirm: "¿Quiere dejar de bloquear esta cuenta?"
|
|||||||
suspendConfirm: "¿Quiere suspender esta cuenta?"
|
suspendConfirm: "¿Quiere suspender esta cuenta?"
|
||||||
unsuspendConfirm: "¿Quiere dejar de suspender esta cuenta?"
|
unsuspendConfirm: "¿Quiere dejar de suspender esta cuenta?"
|
||||||
selectList: "Seleccione una lista"
|
selectList: "Seleccione una lista"
|
||||||
|
selectChannel: "Seleccionar canal"
|
||||||
selectAntenna: "Seleccionar antena"
|
selectAntenna: "Seleccionar antena"
|
||||||
selectWidget: "Seleccionar widget"
|
selectWidget: "Seleccionar widget"
|
||||||
editWidgets: "Editar widgets"
|
editWidgets: "Editar widgets"
|
||||||
@@ -509,7 +510,7 @@ objectStorageSetPublicRead: "Seleccionar \"public-read\" al subir "
|
|||||||
serverLogs: "Registros del servidor"
|
serverLogs: "Registros del servidor"
|
||||||
deleteAll: "Eliminar todos"
|
deleteAll: "Eliminar todos"
|
||||||
showFixedPostForm: "Mostrar el formulario de las entradas encima de la línea de tiempo"
|
showFixedPostForm: "Mostrar el formulario de las entradas encima de la línea de tiempo"
|
||||||
newNoteRecived: "Tienes una nota nuevo"
|
newNoteRecived: "Tienes una nota nueva"
|
||||||
sounds: "Sonidos"
|
sounds: "Sonidos"
|
||||||
sound: "Sonidos"
|
sound: "Sonidos"
|
||||||
listen: "Escuchar"
|
listen: "Escuchar"
|
||||||
@@ -918,17 +919,326 @@ tools: "Utilidades"
|
|||||||
cannotLoad: "No se puede cargar."
|
cannotLoad: "No se puede cargar."
|
||||||
numberOfProfileView: "Número de vistas de perfil"
|
numberOfProfileView: "Número de vistas de perfil"
|
||||||
like: "¡Muy bien!"
|
like: "¡Muy bien!"
|
||||||
|
unlike: "Quitar 'me gusta'"
|
||||||
|
numberOfLikes: "Cantidad de 'Me gusta'"
|
||||||
show: "Apariencia"
|
show: "Apariencia"
|
||||||
|
neverShow: "No mostrar de nuevo"
|
||||||
|
remindMeLater: "Recordar después"
|
||||||
|
didYouLikeMisskey: "¿Te gusta Misskey?"
|
||||||
|
pleaseDonate: "Misskey es software libre, y es usado por {host} . Por favor, ¡considera donar al proyecto principal para que podamos continuar!"
|
||||||
|
roles: "Roles"
|
||||||
|
role: "Roles"
|
||||||
|
normalUser: "Usuario normal"
|
||||||
|
undefined: "Indefinido"
|
||||||
|
assign: "Asignar"
|
||||||
|
unassign: "Quitar"
|
||||||
color: "Color"
|
color: "Color"
|
||||||
|
manageCustomEmojis: "Administrar emojis personalizados"
|
||||||
|
youCannotCreateAnymore: "Se alcanzó el límite de creación"
|
||||||
|
cannotPerformTemporary: "Indisponible temporalmente"
|
||||||
|
cannotPerformTemporaryDescription: "Esta acción no se puede realizar porque se excedió el límite de ejecución. Espera un poco y prueba de nuevo."
|
||||||
|
preset: "Predefinido"
|
||||||
|
selectFromPresets: "Escoger desde predefinidos"
|
||||||
|
achievements: "Logros"
|
||||||
|
_achievements:
|
||||||
|
earnedAt: "Desbloqueado el"
|
||||||
|
_types:
|
||||||
|
_notes1:
|
||||||
|
title: "¡Hola Misskey!"
|
||||||
|
description: "Publicaste tu primera nota"
|
||||||
|
flavor: "¡Pasándola bien con Misskey!"
|
||||||
|
_notes10:
|
||||||
|
title: "Algunas notas"
|
||||||
|
description: "10 notas publicadas"
|
||||||
|
_notes100:
|
||||||
|
title: "¡Muchas notas!"
|
||||||
|
description: "100 notas publicadas"
|
||||||
|
_notes500:
|
||||||
|
title: "¡Cubierto de notas!"
|
||||||
|
description: "500 notas publicadas"
|
||||||
|
_notes1000:
|
||||||
|
title: "¡Una montaña de notas!"
|
||||||
|
description: "1000 notas publicadas"
|
||||||
|
_notes5000:
|
||||||
|
title: "¡Exceso de notas!"
|
||||||
|
description: "5000 notas publicadas"
|
||||||
|
_notes10000:
|
||||||
|
title: "¡Súpernota!"
|
||||||
|
description: "10000 notas publicadas"
|
||||||
|
_notes20000:
|
||||||
|
title: "Necesito... Más... ¡Notas!"
|
||||||
|
description: "20000 notas publicadas"
|
||||||
|
_notes30000:
|
||||||
|
title: "¡Notas! ¡Notas! ¡Notas!"
|
||||||
|
description: "30000 notas publicadas"
|
||||||
|
_notes40000:
|
||||||
|
title: "Fábrica de notas"
|
||||||
|
description: "40000 notas publicadas"
|
||||||
|
_notes50000:
|
||||||
|
title: "¡Un planeta de notas!"
|
||||||
|
description: "50000 notas publicadas"
|
||||||
|
_notes60000:
|
||||||
|
title: "¡Un cuásar de notas!"
|
||||||
|
description: "60000 notas publicadas"
|
||||||
|
_notes70000:
|
||||||
|
title: "¡Un hoyo negro de notas!"
|
||||||
|
description: "70000 notas publicadas"
|
||||||
|
_notes80000:
|
||||||
|
title: "¡Una galaxia de notas!"
|
||||||
|
description: "80000 notas publicadas"
|
||||||
|
_notes90000:
|
||||||
|
title: "¡Todo un universo de notas!"
|
||||||
|
description: "90000 notas publicadas"
|
||||||
|
_notes100000:
|
||||||
|
title: "ALL YOUR NOTE ARE BELONG TO US"
|
||||||
|
description: "100000 notas publicadas"
|
||||||
|
flavor: "¿Tienes tanto para publicar?"
|
||||||
|
_login3:
|
||||||
|
title: "Principiante I"
|
||||||
|
description: "Días desde el inicio de sesión: 3"
|
||||||
|
flavor: "Desde hoy, soy Misskero"
|
||||||
|
_login7:
|
||||||
|
title: "Principiante II"
|
||||||
|
description: "Días desde el inicio de sesión: 7"
|
||||||
|
flavor: "¿Ya te acostumbraste?"
|
||||||
|
_login15:
|
||||||
|
title: "Principiante III"
|
||||||
|
description: "Días desde el inicio de sesión: 15"
|
||||||
|
_login30:
|
||||||
|
title: "Misskero I"
|
||||||
|
description: "Días desde el inicio de sesión: 30"
|
||||||
|
_login60:
|
||||||
|
title: "Misskero II"
|
||||||
|
description: "Días desde el inicio de sesión: 60"
|
||||||
|
_login100:
|
||||||
|
title: "Misskero III"
|
||||||
|
description: "Días desde el inicio de sesión: 100"
|
||||||
|
flavor: "Para este usuario, Misskaína"
|
||||||
|
_login200:
|
||||||
|
title: "Regular I"
|
||||||
|
description: "Días desde el inicio de sesión: 200"
|
||||||
|
_login300:
|
||||||
|
title: "Regular II"
|
||||||
|
description: "Días desde el inicio de sesión: 300"
|
||||||
|
_login400:
|
||||||
|
title: "Regular III"
|
||||||
|
description: "Días desde el inicio de sesión: 400"
|
||||||
|
_login500:
|
||||||
|
title: "Veterano I"
|
||||||
|
description: "Días desde el inicio de sesión: 500"
|
||||||
|
flavor: "Chicos, me encantan las libretas..."
|
||||||
|
_login600:
|
||||||
|
title: "Veterano II"
|
||||||
|
description: "Días desde el inicio de sesión: 600"
|
||||||
|
_login700:
|
||||||
|
title: "Veterano III"
|
||||||
|
description: "Días desde el inicio de sesión: 700"
|
||||||
|
_login800:
|
||||||
|
title: "Maestro I"
|
||||||
|
description: "Días desde el inicio de sesión: 800"
|
||||||
|
_login900:
|
||||||
|
title: "Maestro II"
|
||||||
|
description: "Días desde el inicio de sesión: 900"
|
||||||
|
_login1000:
|
||||||
|
title: "Maestro III"
|
||||||
|
description: "Días desde el inicio de sesión: 1000"
|
||||||
|
flavor: "¡Gracias por usar Misskey!"
|
||||||
|
_noteClipped1:
|
||||||
|
title: "No puedo evitar clipearte..."
|
||||||
|
description: "Hacer un clip por primera vez"
|
||||||
|
_noteFavorited1:
|
||||||
|
title: "Contemplando las estrellas"
|
||||||
|
description: "Poner una nota como favorito por primera vez"
|
||||||
|
_myNoteFavorited1:
|
||||||
|
title: "¡Quiero una estrella!"
|
||||||
|
description: "Tu nota ha sido marcada como favorito por primera vez"
|
||||||
|
_profileFilled:
|
||||||
|
title: "¡Listo!"
|
||||||
|
description: "Perfil completado"
|
||||||
|
_markedAsCat:
|
||||||
|
title: "Soy un gato"
|
||||||
|
description: "Configurar la cuenta como cuenta de un gato"
|
||||||
|
flavor: "Aún no tengo nombre"
|
||||||
|
_following1:
|
||||||
|
title: "Primera vez siguiendo a alguien"
|
||||||
|
description: "Seguir a un usuario"
|
||||||
|
_following10:
|
||||||
|
title: "Ahí la llevas, ahí la llevas..."
|
||||||
|
description: "10 usuarios seguidos"
|
||||||
|
_following50:
|
||||||
|
title: "¡Un puñado de amigos!"
|
||||||
|
description: "50 cuentas seguidas"
|
||||||
|
_following100:
|
||||||
|
title: "100 amigos"
|
||||||
|
description: "100 cuentas seguidas"
|
||||||
|
_following300:
|
||||||
|
title: "¡Sobrecarga de amigos!"
|
||||||
|
description: "300 cuentas seguidas"
|
||||||
|
_followers1:
|
||||||
|
title: "¡Tu primer seguidor!"
|
||||||
|
description: "1 seguidor ganado"
|
||||||
|
_followers10:
|
||||||
|
title: "¡Sígueme!"
|
||||||
|
description: "10 seguidores ganados"
|
||||||
|
_followers50:
|
||||||
|
title: "Viniendo en manada"
|
||||||
|
description: "50 seguidores ganados"
|
||||||
|
_followers100:
|
||||||
|
title: "Popular"
|
||||||
|
description: "100 cuentas seguidas"
|
||||||
|
_followers300:
|
||||||
|
title: "Por favor, hagan una fila"
|
||||||
|
description: "300 seguidores ganados"
|
||||||
|
_followers500:
|
||||||
|
title: "¡Toda una torre de radio!"
|
||||||
|
description: "500 seguidores ganados"
|
||||||
|
_followers1000:
|
||||||
|
title: "\"Influyente\""
|
||||||
|
description: "1000 seguidores gandos"
|
||||||
|
_collectAchievements30:
|
||||||
|
title: "Coleccionista"
|
||||||
|
description: "30 logros ganados"
|
||||||
|
_viewAchievements3min:
|
||||||
|
title: "¡Te gustan los logros!"
|
||||||
|
description: "Mirando tus logros por 3 minutos"
|
||||||
|
_iLoveMisskey:
|
||||||
|
title: "¡AMO Misskey!"
|
||||||
|
description: "\"I ❤ #Misskey\" Publicado"
|
||||||
|
flavor: "El equipo de desarrollo de Misskey, en verdad, ¡aprecia tu apoyo!"
|
||||||
|
_foundTreasure:
|
||||||
|
title: "Búsqueda del tesoro"
|
||||||
|
description: "Encontraste un tesoro"
|
||||||
|
_client30min:
|
||||||
|
title: "Un descansito"
|
||||||
|
description: "30 minutos dedicados a Misskey"
|
||||||
|
_noteDeletedWithin1min:
|
||||||
|
title: "Ah... Mejor no..."
|
||||||
|
description: "Borrar una nota antes que de pase 1 minuto"
|
||||||
|
_postedAtLateNight:
|
||||||
|
title: "Nocturno"
|
||||||
|
description: "Una nota publicada por la noche"
|
||||||
|
flavor: "¡Ya casi es hora de dormir!"
|
||||||
|
_postedAt0min0sec:
|
||||||
|
title: "Reloj parlante"
|
||||||
|
description: "Publicar una nota a las 00:00 de la madrugada"
|
||||||
|
flavor: "Tic, tic, tic ¡TUUUUUN!"
|
||||||
|
_selfQuote:
|
||||||
|
title: "Autoreferencia"
|
||||||
|
description: "Citar tu propia nota"
|
||||||
|
_htl20npm:
|
||||||
|
title: "Línea de tiempo fluyendo"
|
||||||
|
description: "La velocidad de tu línea de tiempo excede las 20 npm (notas por minuto)"
|
||||||
|
_viewInstanceChart:
|
||||||
|
title: "Analista"
|
||||||
|
description: "Gráficas de la instancia mostradas"
|
||||||
|
_outputHelloWorldOnScratchpad:
|
||||||
|
title: "¡Hola mundo!"
|
||||||
|
description: "Escribir \"hello world\" en el compositor"
|
||||||
|
_open3windows:
|
||||||
|
title: "Multiventana"
|
||||||
|
description: "Tener más de 3 ventanas al mismo tiempo"
|
||||||
|
_driveFolderCircularReference:
|
||||||
|
title: "Referencia circular"
|
||||||
|
description: "Intento de crear carpetas recursivamente"
|
||||||
|
_reactWithoutRead:
|
||||||
|
title: "¡Sí lo leíste bien?"
|
||||||
|
description: "Reaccionar a los 3 segundos de publicación de una nota con más de 100 caracteres"
|
||||||
|
_clickedClickHere:
|
||||||
|
title: "Pícale aquí"
|
||||||
|
description: "Le picó ahí"
|
||||||
|
_justPlainLucky:
|
||||||
|
title: "Pura suerte"
|
||||||
|
description: "Obtenido con una probabilidad del 0.01% cada 10 segundos"
|
||||||
|
_setNameToSyuilo:
|
||||||
|
title: "Complejo de superioridad"
|
||||||
|
description: "Configurar el nombre como 'Syuilo'"
|
||||||
|
_passedSinceAccountCreated1:
|
||||||
|
title: "Primer aniversario"
|
||||||
|
description: "Pasó un año desde la creación de la cuenta"
|
||||||
|
_passedSinceAccountCreated2:
|
||||||
|
title: "Segundo aniversario"
|
||||||
|
description: "Pasaron dos años desde la creación de la cuenta"
|
||||||
|
_passedSinceAccountCreated3:
|
||||||
|
title: "Tercer aniversario"
|
||||||
|
description: "Pasaron tres años desde la creación de la cuenta"
|
||||||
|
_loggedInOnBirthday:
|
||||||
|
title: "¡Feliz cumpleaños!"
|
||||||
|
description: "En linea el día de tu cumpleaños"
|
||||||
|
_loggedInOnNewYearsDay:
|
||||||
|
title: "¡Feliz Año Nuevo!"
|
||||||
|
description: "En linea en año nuevo"
|
||||||
|
flavor: "¡Gracias por tu apoyo a la instancia durante todo este año!"
|
||||||
|
_cookieClicked:
|
||||||
|
title: "Un juego para picarle a una galleta"
|
||||||
|
description: "Picaste una galleta"
|
||||||
|
flavor: "¿Está mal este juego?"
|
||||||
|
_brainDiver:
|
||||||
|
title: "Brain Diver"
|
||||||
|
description: "Publicaste un vínculo a \"Brain Diver\""
|
||||||
|
flavor: "Misskey-Misskey La-Tu-Ma"
|
||||||
_role:
|
_role:
|
||||||
|
new: "Crear rol"
|
||||||
|
edit: "Editar rol"
|
||||||
|
name: "Nombre del rol"
|
||||||
|
description: "Descripción del rol"
|
||||||
|
permission: "Permisos del rol"
|
||||||
|
descriptionOfPermission: "<b>Moderador</b> Te permite ejecutar acciones básicas de moderación.\n<b>Administradores</b> puede cambiar todas las configuraciones de la instancia."
|
||||||
|
assignTarget: "Asignar objetivo"
|
||||||
|
descriptionOfAssignTarget: "<b>Manual</b> Para cambiar manualmente lo que se incluye en este rol.\n<b>Condicional</b> configura una condición, y los usuarios que cumplan la condición serán incluídos automáticamente."
|
||||||
|
manual: "manual"
|
||||||
|
conditional: "condicional"
|
||||||
|
condition: "condición"
|
||||||
|
isConditionalRole: "Esto es un rol condicional"
|
||||||
|
isPublic: "Publicar rol"
|
||||||
|
descriptionOfIsPublic: "Cualquiera puede ver los usuarios asignados a este rol. También, el perfil del usuario mostrará este rol."
|
||||||
|
options: "Opción"
|
||||||
|
policies: "Política"
|
||||||
|
baseRole: "Rol base"
|
||||||
|
useBaseValue: "Usar los valores del rol base"
|
||||||
|
chooseRoleToAssign: "Selecciona el rol para asignar"
|
||||||
|
iconUrl: "URL del ícono"
|
||||||
|
asBadge: "Mostrar como emblema"
|
||||||
|
descriptionOfAsBadge: "Este ícono de rol se mostrará a lado del nombre de usuario cuando este rol se encuentre activo."
|
||||||
|
canEditMembersByModerator: "Permitir a los moderadores editar los miembros"
|
||||||
|
descriptionOfCanEditMembersByModerator: "Si se activa, los moderadores, al igual que los administradores, serán capaces de asignar/quitar usuarios a éste rol. Si se desactiva, sólo los administradores podrán hacerlo."
|
||||||
priority: "Prioridad"
|
priority: "Prioridad"
|
||||||
_priority:
|
_priority:
|
||||||
low: "Baja"
|
low: "Baja"
|
||||||
middle: "Mediano"
|
middle: "Mediano"
|
||||||
high: "Alta"
|
high: "Alta"
|
||||||
|
_options:
|
||||||
|
gtlAvailable: "Explorar la línea de tiempo global"
|
||||||
|
ltlAvailable: "Explorar la línea de tiempo local"
|
||||||
|
canPublicNote: "Permitir la publicación"
|
||||||
|
canInvite: "Puede crear códigos de invitación"
|
||||||
|
canManageCustomEmojis: "Administrar emojis personalizados"
|
||||||
|
driveCapacity: "Capacidad de almacenamiento"
|
||||||
|
pinMax: "Máximo de notas fijadas"
|
||||||
|
antennaMax: "Máximo de antenas"
|
||||||
|
wordMuteMax: "Máximo de caracteres en palabras silenciadas"
|
||||||
|
webhookMax: "Máximo de Webhooks"
|
||||||
|
clipMax: "Máximo de clips"
|
||||||
|
noteEachClipsMax: "Máximo de notas con clip"
|
||||||
|
userListMax: "Máximo de listas de usuarios"
|
||||||
|
userEachUserListsMax: "Máximo de usuarios en una lista"
|
||||||
|
rateLimitFactor: "Limitador"
|
||||||
|
descriptionOfRateLimitFactor: "Límites más bajos son menos restrictivos, más altos menos restrictivos"
|
||||||
|
canHideAds: "Puede ocultar anuncios"
|
||||||
|
_condition:
|
||||||
|
isLocal: "Usuario local"
|
||||||
|
isRemote: "Usuario remoto"
|
||||||
|
createdLessThan: "Menos de X han pasado desde la creación de la cuenta"
|
||||||
|
createdMoreThan: "Más de X han pasado desde la creación de la cuenta"
|
||||||
|
followersLessThanOrEq: "Tiene X o menos seguidores"
|
||||||
|
followersMoreThanOrEq: "Tiene X o más seguidores"
|
||||||
|
followingLessThanOrEq: "Sigue X o menos cuentas"
|
||||||
|
followingMoreThanOrEq: "Sigue X o más cuentas"
|
||||||
|
and: "Condicional AND"
|
||||||
|
or: "Condicional OR"
|
||||||
|
not: "Condicional NOT"
|
||||||
_sensitiveMediaDetection:
|
_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."
|
description: "Reduce el esfuerzo de la moderación en 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"
|
sensitivity: "Sensibilidad de la detección"
|
||||||
sensitivityDescription: "Reducir la sensibilidad puede acarrear a varios falsos positivos, mientras que incrementarla puede reducir las detecciones (falsos negativos)."
|
sensitivityDescription: "Reducir la sensibilidad puede acarrear a varios falsos positivos, mientras que incrementarla puede reducir las detecciones (falsos negativos)."
|
||||||
setSensitiveFlagAutomatically: "Marcar como NSFW"
|
setSensitiveFlagAutomatically: "Marcar como NSFW"
|
||||||
setSensitiveFlagAutomaticallyDescription: "Los resultados de la detección interna pueden ser retenidos incluso si la opción está desactivada."
|
setSensitiveFlagAutomaticallyDescription: "Los resultados de la detección interna pueden ser retenidos incluso si la opción está desactivada."
|
||||||
@@ -1328,10 +1638,12 @@ _widgets:
|
|||||||
jobQueue: "Cola de trabajos"
|
jobQueue: "Cola de trabajos"
|
||||||
serverMetric: "Estadísticas del servidor"
|
serverMetric: "Estadísticas del servidor"
|
||||||
aiscript: "Consola de AiScript"
|
aiscript: "Consola de AiScript"
|
||||||
|
aiscriptApp: "Aplicación AiScript"
|
||||||
aichan: "indigo"
|
aichan: "indigo"
|
||||||
userList: "Lista de usuarios"
|
userList: "Lista de usuarios"
|
||||||
_userList:
|
_userList:
|
||||||
chooseList: "Seleccione una lista"
|
chooseList: "Seleccione una lista"
|
||||||
|
clicker: "Cliqueador"
|
||||||
_cw:
|
_cw:
|
||||||
hide: "Ocultar"
|
hide: "Ocultar"
|
||||||
show: "Ver más"
|
show: "Ver más"
|
||||||
@@ -1434,7 +1746,16 @@ _timelines:
|
|||||||
social: "Social"
|
social: "Social"
|
||||||
global: "Global"
|
global: "Global"
|
||||||
_play:
|
_play:
|
||||||
|
new: "Crear guión"
|
||||||
|
edit: "Editar guión"
|
||||||
|
created: "Guión creado"
|
||||||
|
updated: "Guión editado"
|
||||||
|
deleted: "Guión eliminado"
|
||||||
|
pageSetting: "Configuración de guión"
|
||||||
|
editThisPage: "Editar este guión"
|
||||||
viewSource: "Ver la fuente"
|
viewSource: "Ver la fuente"
|
||||||
|
my: "Mis guiones"
|
||||||
|
liked: "Guiones que te gustaron"
|
||||||
featured: "Popular"
|
featured: "Popular"
|
||||||
title: "Título"
|
title: "Título"
|
||||||
script: "Script"
|
script: "Script"
|
||||||
@@ -1507,6 +1828,7 @@ _notification:
|
|||||||
pollEnded: "Estan disponibles los resultados de la encuesta"
|
pollEnded: "Estan disponibles los resultados de la encuesta"
|
||||||
unreadAntennaNote: "Antena {name}"
|
unreadAntennaNote: "Antena {name}"
|
||||||
emptyPushNotificationMessage: "Se han actualizado las notificaciones push"
|
emptyPushNotificationMessage: "Se han actualizado las notificaciones push"
|
||||||
|
achievementEarned: "Logro desbloqueado"
|
||||||
_types:
|
_types:
|
||||||
all: "Todo"
|
all: "Todo"
|
||||||
follow: "Siguiendo"
|
follow: "Siguiendo"
|
||||||
@@ -1548,5 +1870,6 @@ _deck:
|
|||||||
tl: "Linea de tiempo"
|
tl: "Linea de tiempo"
|
||||||
antenna: "Antenas"
|
antenna: "Antenas"
|
||||||
list: "Listas"
|
list: "Listas"
|
||||||
|
channel: "Canal"
|
||||||
mentions: "Menciones"
|
mentions: "Menciones"
|
||||||
direct: "Mensaje directo"
|
direct: "Mensaje directo"
|
||||||
|
@@ -1541,5 +1541,6 @@ _deck:
|
|||||||
tl: "Fil"
|
tl: "Fil"
|
||||||
antenna: "Antennes"
|
antenna: "Antennes"
|
||||||
list: "Listes"
|
list: "Listes"
|
||||||
|
channel: "Canaux"
|
||||||
mentions: "Mentions"
|
mentions: "Mentions"
|
||||||
direct: "Direct"
|
direct: "Direct"
|
||||||
|
@@ -13,6 +13,7 @@ fetchingAsApObject: "Mengambil data dari Fediverse..."
|
|||||||
ok: "OK"
|
ok: "OK"
|
||||||
gotIt: "Saya mengerti"
|
gotIt: "Saya mengerti"
|
||||||
cancel: "Batalkan"
|
cancel: "Batalkan"
|
||||||
|
noThankYou: "Tidak sekarang."
|
||||||
enterUsername: "Masukkan nama pengguna"
|
enterUsername: "Masukkan nama pengguna"
|
||||||
renotedBy: "direnote oleh {user}"
|
renotedBy: "direnote oleh {user}"
|
||||||
noNotes: "Tidak ada catatan"
|
noNotes: "Tidak ada catatan"
|
||||||
@@ -206,6 +207,7 @@ done: "Selesai"
|
|||||||
processing: "Memproses"
|
processing: "Memproses"
|
||||||
preview: "Pratinjau"
|
preview: "Pratinjau"
|
||||||
default: "Bawaan"
|
default: "Bawaan"
|
||||||
|
defaultValueIs: "Bawaan: {value}"
|
||||||
noCustomEmojis: "Tidak ada emoji kustom"
|
noCustomEmojis: "Tidak ada emoji kustom"
|
||||||
noJobs: "Tidak ada kerja"
|
noJobs: "Tidak ada kerja"
|
||||||
federating: "memfederasi"
|
federating: "memfederasi"
|
||||||
@@ -349,6 +351,8 @@ recaptcha: "reCAPTCHA"
|
|||||||
enableRecaptcha: "Nyalakan reCAPTCHA"
|
enableRecaptcha: "Nyalakan reCAPTCHA"
|
||||||
recaptchaSiteKey: "Site key"
|
recaptchaSiteKey: "Site key"
|
||||||
recaptchaSecretKey: "Secret Key"
|
recaptchaSecretKey: "Secret Key"
|
||||||
|
turnstile: "Turnstile"
|
||||||
|
enableTurnstile: "Nyalakan Turnstile"
|
||||||
turnstileSiteKey: "Site key"
|
turnstileSiteKey: "Site key"
|
||||||
turnstileSecretKey: "Secret Key"
|
turnstileSecretKey: "Secret Key"
|
||||||
avoidMultiCaptchaConfirm: "Menggunakan banyak Captcha dapat menyebabkan gangguan. Apakah kamu ingin untuk menonaktifkan Captcha yang lain? Kamu dapat membiarkan fitur ini tetap aktif dengan menekan tombol batal."
|
avoidMultiCaptchaConfirm: "Menggunakan banyak Captcha dapat menyebabkan gangguan. Apakah kamu ingin untuk menonaktifkan Captcha yang lain? Kamu dapat membiarkan fitur ini tetap aktif dengan menekan tombol batal."
|
||||||
@@ -454,6 +458,7 @@ uiLanguage: "Bahasa antarmuka pengguna"
|
|||||||
groupInvited: "Telah diundang ke grup"
|
groupInvited: "Telah diundang ke grup"
|
||||||
aboutX: "Tentang {x}"
|
aboutX: "Tentang {x}"
|
||||||
emojiStyle: "Gaya emoji"
|
emojiStyle: "Gaya emoji"
|
||||||
|
native: "Native"
|
||||||
disableDrawer: "Jangan gunakan menu bergaya laci"
|
disableDrawer: "Jangan gunakan menu bergaya laci"
|
||||||
youHaveNoGroups: "Kamu tidak memiliki grup"
|
youHaveNoGroups: "Kamu tidak memiliki grup"
|
||||||
joinOrCreateGroup: "Bergabunglah dengan grup atau kamu dapat membuat grupmu sendiri."
|
joinOrCreateGroup: "Bergabunglah dengan grup atau kamu dapat membuat grupmu sendiri."
|
||||||
@@ -857,10 +862,21 @@ rateLimitExceeded: "Batas sudah terlampaui"
|
|||||||
cropImage: "potong gambar"
|
cropImage: "potong gambar"
|
||||||
cropImageAsk: "Ingin memotong gambar?"
|
cropImageAsk: "Ingin memotong gambar?"
|
||||||
file: "Berkas"
|
file: "Berkas"
|
||||||
|
recentNHours: "{n} jam terakhir"
|
||||||
|
recentNDays: "{n} hari terakhir"
|
||||||
noEmailServerWarning: "Mail Server tidak disetel."
|
noEmailServerWarning: "Mail Server tidak disetel."
|
||||||
|
thereIsUnresolvedAbuseReportWarning: "Ada laporan yang belum diselesaikan."
|
||||||
recommended: "Disarankan"
|
recommended: "Disarankan"
|
||||||
check: "Cek"
|
check: "Cek"
|
||||||
|
driveCapOverrideLabel: "Ubah kapasitas drive untuk user ini"
|
||||||
|
driveCapOverrideCaption: "Setel ulang kapasitas ke bawaan dengan memasukkan nilai 0 atau lebih rendah."
|
||||||
|
requireAdminForView: "Kamu harus login dengan akun administrator untuk melihat ini."
|
||||||
|
isSystemAccount: "Akun yang dibuat dan otomatis dioperasikan oleh sistem."
|
||||||
|
typeToConfirm: "Mohon masukkan {x} untuk mengonfirmasi"
|
||||||
deleteAccount: "Hapus Akun"
|
deleteAccount: "Hapus Akun"
|
||||||
|
document: "Dokumen"
|
||||||
|
numberOfPageCache: "Jumlah halaman ditembolokkan"
|
||||||
|
numberOfPageCacheDescription: "Menaikkan jumlah ini akan meningkatkan kenyamanan untuk pengguna, namun dapat menyebabkan lonjakan beban pada peladen dan juga memori yang digunakan."
|
||||||
logoutConfirm: "Anda yakin ingin keluar?"
|
logoutConfirm: "Anda yakin ingin keluar?"
|
||||||
lastActiveDate: "Terakhir digunakan"
|
lastActiveDate: "Terakhir digunakan"
|
||||||
statusbar: "Bilah status"
|
statusbar: "Bilah status"
|
||||||
@@ -870,20 +886,189 @@ colored: "Diwarnai"
|
|||||||
refreshInterval: "Jeda pembaharuan"
|
refreshInterval: "Jeda pembaharuan"
|
||||||
label: "Label"
|
label: "Label"
|
||||||
type: "Tipe"
|
type: "Tipe"
|
||||||
|
speed: "Kecepatan"
|
||||||
|
slow: "Lambat"
|
||||||
|
fast: "Cepat"
|
||||||
|
sensitiveMediaDetection: "Deteksi media NSFW"
|
||||||
localOnly: "Hanya lokal"
|
localOnly: "Hanya lokal"
|
||||||
|
remoteOnly: "Hanya remot"
|
||||||
|
failedToUpload: "Gagal mengunggah"
|
||||||
|
cannotUploadBecauseInappropriate: "Berkas ini tidak dapat diunggah karena sebagian dari berkas terdeteksi berpotensi NSFW."
|
||||||
|
cannotUploadBecauseNoFreeSpace: "Gagal mengunggah karena kekurangan kapasitas Drive."
|
||||||
|
beta: "Beta"
|
||||||
|
enableAutoSensitive: "Penandaan NSFW otomatis"
|
||||||
|
enableAutoSensitiveDescription: "Mendeteksi otomatis dan menandai media NSFW menggunakan Machine Learning jika memungkinkan. Meskipun opsi ini dimatikan, ada kemungkinan dinyalakan secara menyeluruh pada instansi peladen."
|
||||||
|
activeEmailValidationDescription: "Membolehkan validasi alamat surel ketat dengan mengecek apakah alamat surel tersebut temporer dan bisa berkomunikasi dengan surel tersebut. Ketidak tidak dicentang, hanya format surel yang divalidasi."
|
||||||
|
navbar: "Bilah navigasi"
|
||||||
shuffle: "Acak"
|
shuffle: "Acak"
|
||||||
account: "Akun"
|
account: "Akun"
|
||||||
|
move: "Pindah"
|
||||||
|
pushNotification: "Pemberitahuan push"
|
||||||
|
subscribePushNotification: "Nyalakan pemberitahuan push"
|
||||||
|
unsubscribePushNotification: "Matikan pemberitahuan push"
|
||||||
|
pushNotificationAlreadySubscribed: "Pemberitahuan push telah dinyalakan"
|
||||||
|
pushNotificationNotSupported: "Browser atau instansi kamu tidak mendukung pemberitahuan push"
|
||||||
|
sendPushNotificationReadMessage: "Hapus pemberitahuan push ketika pemberitahuan relevan atau pesan telah dibaca"
|
||||||
|
sendPushNotificationReadMessageCaption: "Pemberitahuan berisi teks「{emptyPushNotificationMessage}」akan ditampilkan dalam waktu pendek. Ini mungkin dapat menambah pemakaian baterai pada perangkat kamu."
|
||||||
|
windowMaximize: "Maksimalkan"
|
||||||
|
windowRestore: "Kembalikan"
|
||||||
|
caption: "Keterangan"
|
||||||
|
loggedInAsBot: "Sedang login sebagai bot"
|
||||||
|
tools: "Alat"
|
||||||
|
cannotLoad: "Tidak dapat memuat"
|
||||||
|
numberOfProfileView: "tayang profil"
|
||||||
like: "Suka"
|
like: "Suka"
|
||||||
unlike: "Tidak Suka"
|
unlike: "Tidak Suka"
|
||||||
numberOfLikes: "Jumlah yang disukai"
|
numberOfLikes: "Jumlah yang disukai"
|
||||||
show: "Tampilkan"
|
show: "Tampilkan"
|
||||||
|
neverShow: "Jangan tampilkan lagi"
|
||||||
|
remindMeLater: "Mungkin nanti"
|
||||||
|
didYouLikeMisskey: "Apakah kamu mulai menyukai Misskey?"
|
||||||
|
pleaseDonate: "{host} menggunakan perangkat lunak bebas yaitu Misskey. Kami sangat mengapresiasi sekali donasi dari kamu agar pengembangan Misskey tetap dapat berlanjut!"
|
||||||
|
roles: "Peran"
|
||||||
|
role: "Peran"
|
||||||
color: "Warna"
|
color: "Warna"
|
||||||
|
_achievements:
|
||||||
|
_types:
|
||||||
|
_login7:
|
||||||
|
description: "Login selama 7 hari"
|
||||||
|
flavor: "Sudah mulai terbiasa?"
|
||||||
|
_login15:
|
||||||
|
title: "Pemula III"
|
||||||
|
description: "Login selama 15 hari"
|
||||||
|
_login30:
|
||||||
|
title: "Misskist I"
|
||||||
|
description: "Login selama 30 hari"
|
||||||
|
_login60:
|
||||||
|
title: "Misskist II"
|
||||||
|
description: "Login selama 60 hari"
|
||||||
|
_login100:
|
||||||
|
title: "Misskist III"
|
||||||
|
description: "Login selama 100 hari"
|
||||||
|
flavor: "Violent Misskist"
|
||||||
|
_login200:
|
||||||
|
title: "Reguler I"
|
||||||
|
description: "Login selama 200 hari"
|
||||||
|
_login300:
|
||||||
|
title: "Reguler II"
|
||||||
|
description: "Login selama 300 hari"
|
||||||
|
_login400:
|
||||||
|
title: "Reguler III"
|
||||||
|
description: "Login selama 400 hari"
|
||||||
|
_login500:
|
||||||
|
title: "Veteran I"
|
||||||
|
description: "Login selama 500 hari"
|
||||||
|
flavor: "Kawanku, aku suka catatan."
|
||||||
|
_login600:
|
||||||
|
title: "Veteran II"
|
||||||
|
description: "Login selama 600 hari"
|
||||||
|
_login700:
|
||||||
|
title: "Veteran III"
|
||||||
|
description: "Login selama 700 hari"
|
||||||
|
_login800:
|
||||||
|
title: "Sepuh Catatan I"
|
||||||
|
description: "Login selama 800 hari"
|
||||||
|
_login900:
|
||||||
|
title: "Sepuh Catatan II"
|
||||||
|
description: "Login selama 900 hari"
|
||||||
|
_login1000:
|
||||||
|
title: "Sepuh Catatan III"
|
||||||
|
description: "Login selama 1000 hari"
|
||||||
|
flavor: "Terima kasih telah menggunakan Misskey!"
|
||||||
|
_noteClipped1:
|
||||||
|
title: "Harus... Ngeklip..."
|
||||||
|
description: "Klip catatan pertamamu"
|
||||||
|
_noteFavorited1:
|
||||||
|
title: "Pengamat Bintang"
|
||||||
|
description: "Favoritkan catatan pertamamu"
|
||||||
|
_myNoteFavorited1:
|
||||||
|
title: "Pencari Bintang"
|
||||||
|
description: "Minta orang lain memfavoritkan salah satu catatanmu"
|
||||||
|
_profileFilled:
|
||||||
|
title: "Siap Sedia"
|
||||||
|
description: "Atur profil kamu"
|
||||||
|
_markedAsCat:
|
||||||
|
title: "Aku Seekor Kucing"
|
||||||
|
description: "Tandai akunmu sebagai kucing"
|
||||||
|
flavor: "Aku beri kamu nama nanti"
|
||||||
|
_following1:
|
||||||
|
title: "Ikuti pengguna lain pertamamu"
|
||||||
|
description: "Ikuti pengguna"
|
||||||
|
_following10:
|
||||||
|
title: "Terusin... terusin..."
|
||||||
|
description: "Ikuti 10 pengguna lain"
|
||||||
|
_following50:
|
||||||
|
title: "Banyak teman"
|
||||||
|
description: "Ikuti 50 pengguna lain"
|
||||||
|
_following100:
|
||||||
|
title: "100 Teman"
|
||||||
|
description: "Ikuti 100 pengguna lain"
|
||||||
|
_clickedClickHere:
|
||||||
|
description: "Kamu telah mengeklik disini"
|
||||||
|
_justPlainLucky:
|
||||||
|
title: "Lagi Beruntung"
|
||||||
|
description: "Mendapatkan kesempatan dengan kemungkinan 0.01% setiap 10 detik"
|
||||||
|
_setNameToSyuilo:
|
||||||
|
title: "God Complex"
|
||||||
|
description: "Atur namamu jadi \"syuilo\""
|
||||||
|
_passedSinceAccountCreated1:
|
||||||
|
title: "Perayaan Satu Tahun"
|
||||||
|
description: "Satu tahun telah lewat sejak akunmu dibuat"
|
||||||
|
_passedSinceAccountCreated2:
|
||||||
|
title: "Perayaan Dua Tahun"
|
||||||
|
description: "Dua tahun telah lewat sejak akunmu dibuat"
|
||||||
|
_passedSinceAccountCreated3:
|
||||||
|
title: "Perayaan Tiga Tahun"
|
||||||
|
description: "Tiga tahun telah lewat sejak akunmu dibuat"
|
||||||
|
_loggedInOnBirthday:
|
||||||
|
title: "Selamat Ulang Tahun"
|
||||||
|
description: "Login di hari ulang tahunmu"
|
||||||
|
_loggedInOnNewYearsDay:
|
||||||
|
title: "Selamat Tahun Baru!"
|
||||||
|
description: "Login di hari pertama tahun baru"
|
||||||
|
_cookieClicked:
|
||||||
|
title: "Permainan dimana kamu mengeklik kue"
|
||||||
|
description: "Mengeklik kue"
|
||||||
|
flavor: "Tunggu, apakah kamu sedang berada di website yang benar?"
|
||||||
|
_brainDiver:
|
||||||
|
title: "Brain Diver"
|
||||||
|
description: "Posting tautan mengenai Brain Diver"
|
||||||
|
flavor: "Misskey-Misskey La-Tu-Ma"
|
||||||
_role:
|
_role:
|
||||||
|
new: "Buat peran"
|
||||||
|
edit: "Sunting peran"
|
||||||
|
name: "Nama peran"
|
||||||
|
description: "Deskripsi peran"
|
||||||
|
permission: "Perijinan peran"
|
||||||
|
descriptionOfPermission: "<b>Moderator</b> dapat melakukan operasi moderasi dasar.\n<b>Administrator</b> dapat mengubah seluruh pengaturan instansi."
|
||||||
|
assignTarget: "Tipe tugas"
|
||||||
|
descriptionOfAssignTarget: "<b>Manual</b> untuk mengganti secara manual siapa yang mendapatkan peran ini dan siapa yang tidak.\n<b>Kondisional</b> untuk pengguna secara otomatis dimasukkan atau dihapus dari peran berdasarkan kondisi yang ditentukan."
|
||||||
|
manual: "Manual"
|
||||||
|
conditional: "Kondisional"
|
||||||
|
condition: "Kondisi"
|
||||||
|
isConditionalRole: "Ini adalah peran kondisional"
|
||||||
|
isPublic: "Publikkan Peran"
|
||||||
|
descriptionOfIsPublic: "Siapapun dapat melihat daftar pengguna yang ditugaskan pada peran ini. Tambahan juga peran ini akan ditampilkan ke dalam profil pengguna tentang peran yang ditugaskan."
|
||||||
|
options: "Opsi peran"
|
||||||
|
policies: "Kebijakan"
|
||||||
|
baseRole: "Templat peran"
|
||||||
|
useBaseValue: "Gunakan nilai templat peran"
|
||||||
|
chooseRoleToAssign: "Pilih peran yang ditugaskan"
|
||||||
|
canEditMembersByModerator: "Perbolehkan moderator untuk menyunting daftar anggota untuk peran ini"
|
||||||
|
descriptionOfCanEditMembersByModerator: "Ketika dinyalakan, moderator beserta administrator dapat menugaskan ataupun mencabut pengguna ke peran ini. Ketika dimatikan, hanya administrator saja yang dapat menugaskan pengguna ke peran ini."
|
||||||
priority: "Prioritas"
|
priority: "Prioritas"
|
||||||
_priority:
|
_priority:
|
||||||
low: "Rendah"
|
low: "Rendah"
|
||||||
middle: "Sedang"
|
middle: "Sedang"
|
||||||
high: "Tinggi"
|
high: "Tinggi"
|
||||||
|
_options:
|
||||||
|
gtlAvailable: "Dapat melihat linimasa global"
|
||||||
|
ltlAvailable: "Dapat melihat linimasa lokal"
|
||||||
|
canPublicNote: "Dapat mengirim catatan publik"
|
||||||
|
canInvite: "Dapat membuat kode undangan instansi"
|
||||||
|
canManageCustomEmojis: "Dapat mengelola Emoji kustom"
|
||||||
|
driveCapacity: "Kapasitas Drive"
|
||||||
|
pinMax: "Jumlah maksimal catatan yang disematkan"
|
||||||
_emailUnavailable:
|
_emailUnavailable:
|
||||||
used: "Alamat surel ini telah digunakan"
|
used: "Alamat surel ini telah digunakan"
|
||||||
format: "Format tidak valid."
|
format: "Format tidak valid."
|
||||||
@@ -1167,6 +1352,7 @@ _tutorial:
|
|||||||
step7_1: "Yay, Selamat! Kamu sudah menyelesaikan tutorial dasar Misskey."
|
step7_1: "Yay, Selamat! Kamu sudah menyelesaikan tutorial dasar Misskey."
|
||||||
step7_2: "Jika kamu ingin mempelajari lebih lanjut tentang Misskey, cobalah berkunjung ke bagian {help}."
|
step7_2: "Jika kamu ingin mempelajari lebih lanjut tentang Misskey, cobalah berkunjung ke bagian {help}."
|
||||||
step7_3: "Semoga berhasil dan bersenang-senanglah! 🚀"
|
step7_3: "Semoga berhasil dan bersenang-senanglah! 🚀"
|
||||||
|
step8_3: "Kamu dapat mengganti pengaturan ini nanti."
|
||||||
_2fa:
|
_2fa:
|
||||||
alreadyRegistered: "Kamu telah mendaftarkan perangkat otentikasi dua faktor."
|
alreadyRegistered: "Kamu telah mendaftarkan perangkat otentikasi dua faktor."
|
||||||
registerDevice: "Daftarkan perangkat baru"
|
registerDevice: "Daftarkan perangkat baru"
|
||||||
@@ -1241,10 +1427,13 @@ _widgets:
|
|||||||
trends: "Tren"
|
trends: "Tren"
|
||||||
clock: "Jam"
|
clock: "Jam"
|
||||||
rss: "Pembaca RSS"
|
rss: "Pembaca RSS"
|
||||||
|
rssTicker: "RSS-Ticker"
|
||||||
activity: "Aktivitas"
|
activity: "Aktivitas"
|
||||||
photos: "Foto"
|
photos: "Foto"
|
||||||
digitalClock: "Jam digital"
|
digitalClock: "Jam digital"
|
||||||
|
unixClock: "Jam UNIX"
|
||||||
federation: "Federasi"
|
federation: "Federasi"
|
||||||
|
instanceCloud: "Instansi awan"
|
||||||
postForm: "Buat catatan"
|
postForm: "Buat catatan"
|
||||||
slideshow: "Slideshow"
|
slideshow: "Slideshow"
|
||||||
button: "Tombol"
|
button: "Tombol"
|
||||||
@@ -1254,8 +1443,10 @@ _widgets:
|
|||||||
aiscript: "Konsol AiScript"
|
aiscript: "Konsol AiScript"
|
||||||
aiscriptApp: "Aplikasi AiScript"
|
aiscriptApp: "Aplikasi AiScript"
|
||||||
aichan: "Ai"
|
aichan: "Ai"
|
||||||
|
userList: "Daftar pengguna"
|
||||||
_userList:
|
_userList:
|
||||||
chooseList: "Pilih daftar"
|
chooseList: "Pilih daftar"
|
||||||
|
clicker: "Pengeklik"
|
||||||
_cw:
|
_cw:
|
||||||
hide: "Sembunyikan"
|
hide: "Sembunyikan"
|
||||||
show: "Lihat konten"
|
show: "Lihat konten"
|
||||||
@@ -1319,6 +1510,7 @@ _profile:
|
|||||||
changeBanner: "Ubah header"
|
changeBanner: "Ubah header"
|
||||||
_exportOrImport:
|
_exportOrImport:
|
||||||
allNotes: "Semua catatan"
|
allNotes: "Semua catatan"
|
||||||
|
favoritedNotes: "Catatan favorit"
|
||||||
followingList: "Ikuti"
|
followingList: "Ikuti"
|
||||||
muteList: "Bisukan"
|
muteList: "Bisukan"
|
||||||
blockingList: "Blokir"
|
blockingList: "Blokir"
|
||||||
@@ -1437,7 +1629,9 @@ _notification:
|
|||||||
yourFollowRequestAccepted: "Permintaan mengikuti kamu telah diterima"
|
yourFollowRequestAccepted: "Permintaan mengikuti kamu telah diterima"
|
||||||
youWereInvitedToGroup: "Telah diundang ke grup"
|
youWereInvitedToGroup: "Telah diundang ke grup"
|
||||||
pollEnded: "Hasil Kuesioner telah keluar"
|
pollEnded: "Hasil Kuesioner telah keluar"
|
||||||
|
unreadAntennaNote: "Antena {name}"
|
||||||
emptyPushNotificationMessage: "Pembaruan notifikasi dorong"
|
emptyPushNotificationMessage: "Pembaruan notifikasi dorong"
|
||||||
|
achievementEarned: "Pencapaian didapatkan"
|
||||||
_types:
|
_types:
|
||||||
all: "Semua"
|
all: "Semua"
|
||||||
follow: "Ikuti"
|
follow: "Ikuti"
|
||||||
@@ -1459,6 +1653,7 @@ _deck:
|
|||||||
alwaysShowMainColumn: "Selalu tampilkan kolom utama"
|
alwaysShowMainColumn: "Selalu tampilkan kolom utama"
|
||||||
columnAlign: "Luruskan kolom"
|
columnAlign: "Luruskan kolom"
|
||||||
addColumn: "Tambahkan kolom"
|
addColumn: "Tambahkan kolom"
|
||||||
|
configureColumn: "Atur kolom"
|
||||||
swapLeft: "Pindah ke kiri"
|
swapLeft: "Pindah ke kiri"
|
||||||
swapRight: "Pindah ke kanan"
|
swapRight: "Pindah ke kanan"
|
||||||
swapUp: "Pindah ke atas"
|
swapUp: "Pindah ke atas"
|
||||||
@@ -1466,6 +1661,11 @@ _deck:
|
|||||||
stackLeft: "Tumpukkan di kolom kiri"
|
stackLeft: "Tumpukkan di kolom kiri"
|
||||||
popRight: "Keluarkan di kanan"
|
popRight: "Keluarkan di kanan"
|
||||||
profile: "Profil"
|
profile: "Profil"
|
||||||
|
newProfile: "Profil baru"
|
||||||
|
deleteProfile: "Hapus profil"
|
||||||
|
introduction: "Buat antarmuka sempurna untukmu dengan menata kolom secara bebas!"
|
||||||
|
introduction2: "Klik \"+\" pada kanan layar untuk menambahkan kolom baru kapanpun yang kamu mau."
|
||||||
|
widgetsIntroduction: "Mohon pilih \"Sunting gawit\" pada menu kolom dan tambahkan gawit."
|
||||||
_columns:
|
_columns:
|
||||||
main: "Utama"
|
main: "Utama"
|
||||||
widgets: "Widget"
|
widgets: "Widget"
|
||||||
@@ -1473,5 +1673,6 @@ _deck:
|
|||||||
tl: "Linimasa"
|
tl: "Linimasa"
|
||||||
antenna: "Antena"
|
antenna: "Antena"
|
||||||
list: "Daftar"
|
list: "Daftar"
|
||||||
|
channel: "Kanal"
|
||||||
mentions: "Sebutan"
|
mentions: "Sebutan"
|
||||||
direct: "Langsung"
|
direct: "Langsung"
|
||||||
|
@@ -34,6 +34,7 @@ const languages = [
|
|||||||
'pt-PT',
|
'pt-PT',
|
||||||
'ru-RU',
|
'ru-RU',
|
||||||
'sk-SK',
|
'sk-SK',
|
||||||
|
'th-TH',
|
||||||
'ug-CN',
|
'ug-CN',
|
||||||
'uk-UA',
|
'uk-UA',
|
||||||
'vi-VN',
|
'vi-VN',
|
||||||
|
@@ -1,7 +1,7 @@
|
|||||||
---
|
---
|
||||||
_lang_: "Italiano"
|
_lang_: "Italiano"
|
||||||
headlineMisskey: "Rete collegata tramite note"
|
headlineMisskey: "Rete collegata tramite note"
|
||||||
introMisskey: "Eccoci! Misskey è un servizio di microblogging decentralizzato, libero e aperto. \n📡 Puoi pubblicare «Note» per condividere ciò che sta succedendo o per dire a tutti qualcosa su di te. \n👍 Puoi reagire inviando emoji rapidi alle «Note» provenienti da altri profili nel Fediverso.\n🚀 Esplora un nuovo mondo insieme a noi!"
|
introMisskey: "Eccoci! Misskey è un servizio di microblogging decentralizzato, libero e aperto. \n\n📡 Puoi pubblicare «Note» per condividere ciò che sta succedendo o per dire a tutti qualcosa su di te. \n\n👍 Puoi reagire inviando emoji rapidi alle «Note» provenienti da altri profili nel Fediverso.\n\n🚀 Esplora un nuovo mondo insieme a noi!"
|
||||||
poweredByMisskeyDescription: "{name} è uno dei servizi (chiamati istanze) che utilizzano la piattaforma open source <b>Misskey</b>."
|
poweredByMisskeyDescription: "{name} è uno dei servizi (chiamati istanze) che utilizzano la piattaforma open source <b>Misskey</b>."
|
||||||
monthAndDay: "{day}/{month}"
|
monthAndDay: "{day}/{month}"
|
||||||
search: "Cerca"
|
search: "Cerca"
|
||||||
@@ -95,7 +95,7 @@ follow: "Segui"
|
|||||||
followRequest: "Richiesta di follow"
|
followRequest: "Richiesta di follow"
|
||||||
followRequests: "Richieste di follow"
|
followRequests: "Richieste di follow"
|
||||||
unfollow: "Smetti di seguire"
|
unfollow: "Smetti di seguire"
|
||||||
followRequestPending: "La richiesta di follow deve essere approvata"
|
followRequestPending: "Richiesta in approvazione"
|
||||||
enterEmoji: "Inserisci emoji"
|
enterEmoji: "Inserisci emoji"
|
||||||
renote: "Rinota"
|
renote: "Rinota"
|
||||||
unrenote: "Annulla rinota"
|
unrenote: "Annulla rinota"
|
||||||
@@ -943,79 +943,79 @@ _achievements:
|
|||||||
earnedAt: "Data di conseguimento"
|
earnedAt: "Data di conseguimento"
|
||||||
_types:
|
_types:
|
||||||
_notes1:
|
_notes1:
|
||||||
title: "Ho iniziato a usare Misskey"
|
title: "Hai iniziato a usare Misskey"
|
||||||
description: "Ho pubblicato la mia prima Nota"
|
description: "Hai pubblicato la prima Nota"
|
||||||
flavor: "Goditi la vita su Misskey!"
|
flavor: "Goditi la vita su Misskey!"
|
||||||
_notes10:
|
_notes10:
|
||||||
title: "Alcune Note"
|
title: "Alcune Note"
|
||||||
description: "Ho inserito 10 Note"
|
description: "Hai inserito 10 Note"
|
||||||
_notes100:
|
_notes100:
|
||||||
title: "Un po' di Note"
|
title: "Un po' di Note"
|
||||||
description: "Ho inserito 100 Note"
|
description: "Hai inserito 100 Note"
|
||||||
_notes500:
|
_notes500:
|
||||||
title: "Un bel po' di Note"
|
title: "Un bel po' di Note"
|
||||||
description: "Ho inserito 500 Note"
|
description: "Hai inserito 500 Note"
|
||||||
_notes1000:
|
_notes1000:
|
||||||
title: "Una montagna di Note"
|
title: "Una montagna di Note"
|
||||||
description: "Ho inserito 1.000 Note"
|
description: "Hai inserito 1.000 Note"
|
||||||
_notes5000:
|
_notes5000:
|
||||||
title: "Un sovraccarico di Note!"
|
title: "Un sovraccarico di Note!"
|
||||||
description: "Ho inserito 5.000 Note"
|
description: "Hai inserito 5.000 Note"
|
||||||
_notes10000:
|
_notes10000:
|
||||||
title: "SuperNote!"
|
title: "SuperNote!"
|
||||||
description: "Ho inserito 10.000 Note"
|
description: "Hai inserito 10.000 Note"
|
||||||
_notes20000:
|
_notes20000:
|
||||||
title: "Voglio più... Note!"
|
title: "Voglio più... Note!"
|
||||||
description: "Ho inserito 20.000 Note"
|
description: "Hai inserito 20.000 Note"
|
||||||
_notes30000:
|
_notes30000:
|
||||||
title: "Note, Note, Note!"
|
title: "Note, Note, Note!"
|
||||||
description: "Ho inserito 30.000 Note"
|
description: "Hai inserito 30.000 Note"
|
||||||
_notes40000:
|
_notes40000:
|
||||||
title: "Una fabbrica di Note"
|
title: "Una fabbrica di Note"
|
||||||
description: "Ho inserito 40.000 Note"
|
description: "Hai inserito 40.000 Note"
|
||||||
_notes50000:
|
_notes50000:
|
||||||
title: "Un pianeta di Note"
|
title: "Un pianeta di Note"
|
||||||
description: "Ho inserito 50.000 Note"
|
description: "Hai inserito 50.000 Note"
|
||||||
_notes60000:
|
_notes60000:
|
||||||
title: "Un quasar di Note"
|
title: "Un quasar di Note"
|
||||||
description: "Ho inserito 60.000 Note"
|
description: "Hai inserito 60.000 Note"
|
||||||
_notes70000:
|
_notes70000:
|
||||||
title: "Un buco nero supermassiccio di Note"
|
title: "Un buco nero supermassiccio di Note"
|
||||||
description: "Ho inserito 70.000 Note"
|
description: "Hai inserito 70.000 Note"
|
||||||
_notes80000:
|
_notes80000:
|
||||||
title: "Una galassia di Note"
|
title: "Una galassia di Note"
|
||||||
description: "Ho inserito 80.000 Note"
|
description: "Hai inserito 80.000 Note"
|
||||||
_notes90000:
|
_notes90000:
|
||||||
title: "Un universo di Note!"
|
title: "Un universo di Note!"
|
||||||
description: "Ho inserito 90.000 Note"
|
description: "Hai inserito 90.000 Note"
|
||||||
_notes100000:
|
_notes100000:
|
||||||
title: "ALL YOUR NOTE ARE BELONG TO US"
|
title: "ALL YOUR NOTE ARE BELONG TO US"
|
||||||
description: "Ho inserito 100.000 Note"
|
description: "Hai inserito 100.000 Note"
|
||||||
flavor: "Hai molto da scrivere?"
|
flavor: "Hai molto da scrivere?"
|
||||||
_login3:
|
_login3:
|
||||||
title: "Principiante I"
|
title: "Principiante I"
|
||||||
description: "Accedi per 3 giorni di fila"
|
description: "Accedi per un totale di 3 giorni"
|
||||||
flavor: "Da oggi, chiamatemi Misskist"
|
flavor: "Da oggi, chiamatemi Misskist"
|
||||||
_login7:
|
_login7:
|
||||||
title: "Principiante II"
|
title: "Principiante II"
|
||||||
description: "Accedi per 7 giorni di fila"
|
description: "Accedi per un totale di 7 giorni"
|
||||||
flavor: "Ti sembra di avere la situazione sotto controllo?"
|
flavor: "Ti sembra di avere la situazione sotto controllo?"
|
||||||
_login15:
|
_login15:
|
||||||
title: "Principiante III"
|
title: "Principiante III"
|
||||||
description: "Accedi per 15 giorni di fila"
|
description: "Accedi per un totale di 15 giorni"
|
||||||
_login30:
|
_login30:
|
||||||
title: "Misskist I"
|
title: "Misskist I"
|
||||||
description: "Accedi per 30 giorni di fila"
|
description: "Accedi per un totale di 30 giorni"
|
||||||
_login60:
|
_login60:
|
||||||
title: "Misskeist II"
|
title: "Misskeist II"
|
||||||
description: "Accedi per 60 giorni di fila"
|
description: "Accedi per un totale di 60 giorni"
|
||||||
_login100:
|
_login100:
|
||||||
title: "Misskeist III"
|
title: "Misskeist III"
|
||||||
description: "Accedi per 100 giorni di fila"
|
description: "Accedi per un totale di 100 giorni"
|
||||||
flavor: "Violent Misskeist"
|
flavor: "Violent Misskeist"
|
||||||
_login200:
|
_login200:
|
||||||
title: "Regolare I"
|
title: "Regolare I"
|
||||||
description: "Accedi per 200 giorni totali"
|
description: "Accedi per un totale di 200 giorni"
|
||||||
_login300:
|
_login300:
|
||||||
title: "Regolare II"
|
title: "Regolare II"
|
||||||
description: "Accedi per un totale di 300 giorni"
|
description: "Accedi per un totale di 300 giorni"
|
||||||
@@ -1044,10 +1044,13 @@ _achievements:
|
|||||||
flavor: "Grazie per aver usato Misskey!"
|
flavor: "Grazie per aver usato Misskey!"
|
||||||
_noteClipped1:
|
_noteClipped1:
|
||||||
title: "Devo clippare!"
|
title: "Devo clippare!"
|
||||||
description: "Ho raccolto in Clip la prima Nota"
|
description: "Hai raccolto la tua prima Nota in una Clip"
|
||||||
_noteFavorited1:
|
_noteFavorited1:
|
||||||
title: "Guarda le stelle"
|
title: "Guarda le stelle"
|
||||||
description: "Aggiungi una Nota ai preferiti per la prima volta"
|
description: "Aggiungi una Nota ai preferiti per la prima volta"
|
||||||
|
_myNoteFavorited1:
|
||||||
|
title: "Fornitura stelline"
|
||||||
|
description: "Qualcuno ha preferito una delle tue Note"
|
||||||
_profileFilled:
|
_profileFilled:
|
||||||
title: "Perfettamente"
|
title: "Perfettamente"
|
||||||
description: "Imposta il tuo profilo"
|
description: "Imposta il tuo profilo"
|
||||||
@@ -1056,8 +1059,8 @@ _achievements:
|
|||||||
description: "Aggiungi le orecchie da gatto al tuo profilo"
|
description: "Aggiungi le orecchie da gatto al tuo profilo"
|
||||||
flavor: "Ti chiamerò..."
|
flavor: "Ti chiamerò..."
|
||||||
_following1:
|
_following1:
|
||||||
title: "Hai seguito il tuo primo profilo"
|
title: "Il mio primo Follow"
|
||||||
description: "Il tuo primo profilo Follower"
|
description: "Hai seguito il tuo primo profilo"
|
||||||
_following10:
|
_following10:
|
||||||
title: "Segui, segui!"
|
title: "Segui, segui!"
|
||||||
description: "Hai seguito 10 profili"
|
description: "Hai seguito 10 profili"
|
||||||
@@ -1071,17 +1074,17 @@ _achievements:
|
|||||||
title: "Sovraccarico di amici"
|
title: "Sovraccarico di amici"
|
||||||
description: "Hai seguito 300 profili"
|
description: "Hai seguito 300 profili"
|
||||||
_followers1:
|
_followers1:
|
||||||
title: "Primo Follower"
|
title: "Il primo profilo tuo Follower"
|
||||||
description: "Hai ottenuto un Follower"
|
description: "Hai ottenuto il tuo primo Follower"
|
||||||
_followers10:
|
_followers10:
|
||||||
title: "Follow me!"
|
title: "Follow me!"
|
||||||
description: "Hai ottenuto 10 Follower"
|
description: "Hai ottenuto 10 profili Follower"
|
||||||
_followers50:
|
_followers50:
|
||||||
title: "Follower a frotte"
|
title: "Un gregge di Follower"
|
||||||
description: "Hai ottenuto 50 Follower"
|
description: "Hai ottenuto 50 Follower"
|
||||||
_followers100:
|
_followers100:
|
||||||
title: "Popolare"
|
title: "Popolare"
|
||||||
description: "Hai ottenuto 100 Follower"
|
description: "Hai ottenuto 100 profili Follower"
|
||||||
_followers300:
|
_followers300:
|
||||||
title: "Mettetevi in fila"
|
title: "Mettetevi in fila"
|
||||||
description: "Hai ottenuto 300 Follower"
|
description: "Hai ottenuto 300 Follower"
|
||||||
@@ -1090,9 +1093,88 @@ _achievements:
|
|||||||
description: "Hai ottenuto 500 Follower"
|
description: "Hai ottenuto 500 Follower"
|
||||||
_followers1000:
|
_followers1000:
|
||||||
title: "Influenzer"
|
title: "Influenzer"
|
||||||
|
description: "Hai superato i 1.000 profili Follower"
|
||||||
|
_collectAchievements30:
|
||||||
|
title: "Collezionista di successi"
|
||||||
|
description: "Hai raggiunto 30 obiettivi"
|
||||||
|
_viewAchievements3min:
|
||||||
|
title: "Mi piacciono i risultati"
|
||||||
|
description: "Guarda la tua collezione di obiettivi per almeno 3 minuti"
|
||||||
|
_iLoveMisskey:
|
||||||
|
title: "I LOVE Misskey"
|
||||||
|
description: "Pubblica «I ♥ #Misskey»"
|
||||||
|
flavor: "Grazie per aver utilizzato Misskey! Dal team di sviluppo"
|
||||||
|
_foundTreasure:
|
||||||
|
title: "Caccia al tesoro"
|
||||||
|
description: "Hai trovato un tesoro nascosto"
|
||||||
|
_client30min:
|
||||||
|
title: "Piccola grande pausa"
|
||||||
|
description: "Hai passato più di 30 minuti su Misskey"
|
||||||
|
_noteDeletedWithin1min:
|
||||||
|
title: "Ooops!"
|
||||||
|
description: "Hai eliminato una nota entro un minuto dalla sua pubblicazione"
|
||||||
|
_postedAtLateNight:
|
||||||
|
title: "Biassanot!"
|
||||||
|
description: "Hai pubblicato una nota in tarda notte"
|
||||||
|
flavor: "Andiamo a dormire presto"
|
||||||
|
_postedAt0min0sec:
|
||||||
|
title: "Mezzanotte"
|
||||||
|
description: "Hai pubblicato una Nota a mezzanotte in punto"
|
||||||
|
flavor: "tic, tac, tic, tac! Gong!"
|
||||||
|
_selfQuote:
|
||||||
|
title: "Autoreferenziale"
|
||||||
|
description: "Hai citato una delle tue Note"
|
||||||
|
_htl20npm:
|
||||||
|
title: "Timeline scorrevole"
|
||||||
|
description: "La tua Timeline personale ha superato la velocità di 20 Note orarie (Note al minuto)"
|
||||||
|
_viewInstanceChart:
|
||||||
|
title: "Analista"
|
||||||
|
description: "Visualizza i grafici dell'istanza"
|
||||||
|
_outputHelloWorldOnScratchpad:
|
||||||
|
title: "Hello, world!"
|
||||||
|
description: "Hai scritto «Hello world» nel blocco appunti"
|
||||||
|
_open3windows:
|
||||||
|
title: "Apri le finestre!"
|
||||||
|
description: "Hai aperto almeno 3 finestre contemporaneamente"
|
||||||
|
_driveFolderCircularReference:
|
||||||
|
title: "Riferimento circolare"
|
||||||
|
description: "Hai provato a nidificare in modo ricorsivo le cartelle del Drive"
|
||||||
|
_reactWithoutRead:
|
||||||
|
title: "Hai letto bene?"
|
||||||
|
description: "Hai reagito ad una Nota più lunga di 100 caratteri entro 3 secondi dalla sua pubblicazione"
|
||||||
|
_clickedClickHere:
|
||||||
|
title: "Clicca qui"
|
||||||
|
description: "Hai cliccato qui"
|
||||||
|
_justPlainLucky:
|
||||||
|
title: "Proprio fortunato"
|
||||||
|
description: "Ottenuto con una probabilità dello 0,01% ogni 10 secondi"
|
||||||
|
_setNameToSyuilo:
|
||||||
|
title: "Complesso divino"
|
||||||
|
description: "Hai impostati il tuo nome in «syuilo»"
|
||||||
|
_passedSinceAccountCreated1:
|
||||||
|
title: "Primo Anniversario"
|
||||||
|
description: "È passato un anno da quando hai creato il profilo"
|
||||||
|
_passedSinceAccountCreated2:
|
||||||
|
title: "Secondo Anniversario"
|
||||||
|
description: "Sono passati due anni da quando hai creato il profilo"
|
||||||
|
_passedSinceAccountCreated3:
|
||||||
|
title: "Terzo Anniversario"
|
||||||
|
description: "Sono passati tre anni da quando hai creato il profilo"
|
||||||
|
_loggedInOnBirthday:
|
||||||
|
title: "Buon compleanno!"
|
||||||
|
description: "Hai effettuato l'accesso il giorno del tuo compleanno"
|
||||||
|
_loggedInOnNewYearsDay:
|
||||||
|
title: "Buon anno nuovo!"
|
||||||
|
description: "Hai usato effettuato l'accesso il giorno di capodanno"
|
||||||
|
flavor: "Anche quest'anno, grazie per il tuo continuo supporto a questa istanza"
|
||||||
|
_cookieClicked:
|
||||||
|
title: "Clicca il biscotto"
|
||||||
|
description: "Hai giocato a cliccare il cookie"
|
||||||
|
flavor: "È il sito giusto?"
|
||||||
_brainDiver:
|
_brainDiver:
|
||||||
title: "Brain Diver"
|
title: "Brain Diver"
|
||||||
description: "Pubblica un link a Brain Diver"
|
description: "Pubblica un link a Brain Diver"
|
||||||
|
flavor: "Sulle note di Brain Diver"
|
||||||
_role:
|
_role:
|
||||||
new: "Nuovo ruolo"
|
new: "Nuovo ruolo"
|
||||||
edit: "Modifica ruolo"
|
edit: "Modifica ruolo"
|
||||||
@@ -1113,6 +1195,9 @@ _role:
|
|||||||
baseRole: "Ruolo di base"
|
baseRole: "Ruolo di base"
|
||||||
useBaseValue: "Eredita dal ruolo base"
|
useBaseValue: "Eredita dal ruolo base"
|
||||||
chooseRoleToAssign: "Seleziona il ruolo da assegnare"
|
chooseRoleToAssign: "Seleziona il ruolo da assegnare"
|
||||||
|
iconUrl: "URL dell'icona"
|
||||||
|
asBadge: "Mostra come badge"
|
||||||
|
descriptionOfAsBadge: "Se indicato, accanto al nome utente viene visualizzata l'icona del ruolo."
|
||||||
canEditMembersByModerator: "Anche i Moderatori assegnano profili a questo ruolo"
|
canEditMembersByModerator: "Anche i Moderatori assegnano profili a questo ruolo"
|
||||||
descriptionOfCanEditMembersByModerator: "Se disattivo, potranno farlo solamente gli Amministratori."
|
descriptionOfCanEditMembersByModerator: "Se disattivo, potranno farlo solamente gli Amministratori."
|
||||||
priority: "Priorità"
|
priority: "Priorità"
|
||||||
@@ -1425,8 +1510,8 @@ _sfx:
|
|||||||
channel: "Notifiche di canale"
|
channel: "Notifiche di canale"
|
||||||
_ago:
|
_ago:
|
||||||
future: "Futuro"
|
future: "Futuro"
|
||||||
justNow: "Ora"
|
justNow: "Adesso"
|
||||||
secondsAgo: "{n}s fa"
|
secondsAgo: "{n} sec fa"
|
||||||
minutesAgo: "{n} min fa"
|
minutesAgo: "{n} min fa"
|
||||||
hoursAgo: "{n} ore fa"
|
hoursAgo: "{n} ore fa"
|
||||||
daysAgo: "{n} gg fa"
|
daysAgo: "{n} gg fa"
|
||||||
@@ -1784,5 +1869,6 @@ _deck:
|
|||||||
tl: "Timeline"
|
tl: "Timeline"
|
||||||
antenna: "Antenne"
|
antenna: "Antenne"
|
||||||
list: "Liste"
|
list: "Liste"
|
||||||
|
channel: "Canale"
|
||||||
mentions: "Menzioni"
|
mentions: "Menzioni"
|
||||||
direct: "Diretta"
|
direct: "Diretta"
|
||||||
|
@@ -129,6 +129,7 @@ unblockConfirm: "ブロック解除しますか?"
|
|||||||
suspendConfirm: "凍結しますか?"
|
suspendConfirm: "凍結しますか?"
|
||||||
unsuspendConfirm: "解凍しますか?"
|
unsuspendConfirm: "解凍しますか?"
|
||||||
selectList: "リストを選択"
|
selectList: "リストを選択"
|
||||||
|
selectChannel: "チャンネルを選択"
|
||||||
selectAntenna: "アンテナを選択"
|
selectAntenna: "アンテナを選択"
|
||||||
selectWidget: "ウィジェットを選択"
|
selectWidget: "ウィジェットを選択"
|
||||||
editWidgets: "ウィジェットを編集"
|
editWidgets: "ウィジェットを編集"
|
||||||
@@ -939,6 +940,8 @@ cannotPerformTemporaryDescription: "操作回数が制限を超過するため
|
|||||||
preset: "プリセット"
|
preset: "プリセット"
|
||||||
selectFromPresets: "プリセットから選択"
|
selectFromPresets: "プリセットから選択"
|
||||||
achievements: "実績"
|
achievements: "実績"
|
||||||
|
gotInvalidResponseError: "サーバーの応答が無効です"
|
||||||
|
gotInvalidResponseErrorDescription: "サーバーがダウンまたはメンテナンスしている可能性があります。しばらくしてから再度お試しください。"
|
||||||
|
|
||||||
_achievements:
|
_achievements:
|
||||||
earnedAt: "獲得日時"
|
earnedAt: "獲得日時"
|
||||||
@@ -1049,6 +1052,9 @@ _achievements:
|
|||||||
_noteFavorited1:
|
_noteFavorited1:
|
||||||
title: "星をみるひと"
|
title: "星をみるひと"
|
||||||
description: "初めてノートをお気に入りに登録した"
|
description: "初めてノートをお気に入りに登録した"
|
||||||
|
_myNoteFavorited1:
|
||||||
|
title: "星が欲しい"
|
||||||
|
description: "自分のノートが他の人からお気に入りに登録された"
|
||||||
_profileFilled:
|
_profileFilled:
|
||||||
title: "準備万端"
|
title: "準備万端"
|
||||||
description: "プロフィール設定を行った"
|
description: "プロフィール設定を行った"
|
||||||
@@ -1102,6 +1108,9 @@ _achievements:
|
|||||||
title: "I Love Misskey"
|
title: "I Love Misskey"
|
||||||
description: "\"I ❤ #Misskey\"を投稿した"
|
description: "\"I ❤ #Misskey\"を投稿した"
|
||||||
flavor: "Misskeyを使ってくださりありがとうございます! by 開発チーム"
|
flavor: "Misskeyを使ってくださりありがとうございます! by 開発チーム"
|
||||||
|
_foundTreasure:
|
||||||
|
title: "宝探し"
|
||||||
|
description: "隠されたお宝を発見した"
|
||||||
_client30min:
|
_client30min:
|
||||||
title: "ひとやすみ"
|
title: "ひとやすみ"
|
||||||
description: "クライアントを起動してから30分以上経過した"
|
description: "クライアントを起動してから30分以上経過した"
|
||||||
@@ -1122,6 +1131,9 @@ _achievements:
|
|||||||
_htl20npm:
|
_htl20npm:
|
||||||
title: "流れるTL"
|
title: "流れるTL"
|
||||||
description: "ホームタイムラインの流速が20npmを越す"
|
description: "ホームタイムラインの流速が20npmを越す"
|
||||||
|
_viewInstanceChart:
|
||||||
|
title: "アナリスト"
|
||||||
|
description: "インスタンスのチャートを表示した"
|
||||||
_outputHelloWorldOnScratchpad:
|
_outputHelloWorldOnScratchpad:
|
||||||
title: "Hello, world!"
|
title: "Hello, world!"
|
||||||
description: "スクラッチパッドで hello world を出力した"
|
description: "スクラッチパッドで hello world を出力した"
|
||||||
@@ -1139,7 +1151,7 @@ _achievements:
|
|||||||
description: "ここをクリックした"
|
description: "ここをクリックした"
|
||||||
_justPlainLucky:
|
_justPlainLucky:
|
||||||
title: "単なるラッキー"
|
title: "単なるラッキー"
|
||||||
description: "10秒ごとに0.01%の確率で獲得"
|
description: "10秒ごとに0.005%の確率で獲得"
|
||||||
_setNameToSyuilo:
|
_setNameToSyuilo:
|
||||||
title: "神様コンプレックス"
|
title: "神様コンプレックス"
|
||||||
description: "名前を syuilo に設定した"
|
description: "名前を syuilo に設定した"
|
||||||
@@ -1175,7 +1187,7 @@ _role:
|
|||||||
description: "ロールの説明"
|
description: "ロールの説明"
|
||||||
permission: "ロールの権限"
|
permission: "ロールの権限"
|
||||||
descriptionOfPermission: "<b>モデレーター</b>は基本的なモデレーションに関する操作を行えます。\n<b>管理者</b>はインスタンスの全ての設定を変更できます。"
|
descriptionOfPermission: "<b>モデレーター</b>は基本的なモデレーションに関する操作を行えます。\n<b>管理者</b>はインスタンスの全ての設定を変更できます。"
|
||||||
assignTarget: "アサインターゲット"
|
assignTarget: "アサイン"
|
||||||
descriptionOfAssignTarget: "<b>マニュアル</b>は誰がこのロールに含まれるかを手動で管理します。\n<b>コンディショナル</b>は条件を設定し、それに合致するユーザーが自動で含まれるようになります。"
|
descriptionOfAssignTarget: "<b>マニュアル</b>は誰がこのロールに含まれるかを手動で管理します。\n<b>コンディショナル</b>は条件を設定し、それに合致するユーザーが自動で含まれるようになります。"
|
||||||
manual: "マニュアル"
|
manual: "マニュアル"
|
||||||
conditional: "コンディショナル"
|
conditional: "コンディショナル"
|
||||||
@@ -1188,6 +1200,9 @@ _role:
|
|||||||
baseRole: "ベースロール"
|
baseRole: "ベースロール"
|
||||||
useBaseValue: "ベースロールの値を使用"
|
useBaseValue: "ベースロールの値を使用"
|
||||||
chooseRoleToAssign: "アサインするロールを選択"
|
chooseRoleToAssign: "アサインするロールを選択"
|
||||||
|
iconUrl: "アイコン画像のURL"
|
||||||
|
asBadge: "バッジとして表示"
|
||||||
|
descriptionOfAsBadge: "オンにすると、ユーザー名の横にロールのアイコンが表示されます。"
|
||||||
canEditMembersByModerator: "モデレーターのメンバー編集を許可"
|
canEditMembersByModerator: "モデレーターのメンバー編集を許可"
|
||||||
descriptionOfCanEditMembersByModerator: "オンにすると、管理者に加えてモデレーターもこのロールへユーザーをアサイン/アサイン解除できるようになります。オフにすると管理者のみが行えます。"
|
descriptionOfCanEditMembersByModerator: "オンにすると、管理者に加えてモデレーターもこのロールへユーザーをアサイン/アサイン解除できるようになります。オフにすると管理者のみが行えます。"
|
||||||
priority: "優先度"
|
priority: "優先度"
|
||||||
@@ -1910,5 +1925,6 @@ _deck:
|
|||||||
tl: "タイムライン"
|
tl: "タイムライン"
|
||||||
antenna: "アンテナ"
|
antenna: "アンテナ"
|
||||||
list: "リスト"
|
list: "リスト"
|
||||||
|
channel: "チャンネル"
|
||||||
mentions: "あなた宛て"
|
mentions: "あなた宛て"
|
||||||
direct: "ダイレクト"
|
direct: "ダイレクト"
|
||||||
|
@@ -1628,5 +1628,6 @@ _deck:
|
|||||||
tl: "タイムライン"
|
tl: "タイムライン"
|
||||||
antenna: "アンテナ"
|
antenna: "アンテナ"
|
||||||
list: "リスト"
|
list: "リスト"
|
||||||
|
channel: "チャンネル"
|
||||||
mentions: "あんた宛て"
|
mentions: "あんた宛て"
|
||||||
direct: "ダイレクト"
|
direct: "ダイレクト"
|
||||||
|
@@ -938,13 +938,13 @@ cannotPerformTemporary: "일시적으로 사용할 수 없음"
|
|||||||
cannotPerformTemporaryDescription: "조작 횟수 제한을 초과하여 일시적으로 사용이 불가합니다. 잠시 후 다시 시도해 주세요."
|
cannotPerformTemporaryDescription: "조작 횟수 제한을 초과하여 일시적으로 사용이 불가합니다. 잠시 후 다시 시도해 주세요."
|
||||||
preset: "프리셋"
|
preset: "프리셋"
|
||||||
selectFromPresets: "프리셋에서 선택"
|
selectFromPresets: "프리셋에서 선택"
|
||||||
achievements: "도전과제"
|
achievements: "도전 과제"
|
||||||
_achievements:
|
_achievements:
|
||||||
earnedAt: "달성 일시"
|
earnedAt: "달성 일시"
|
||||||
_types:
|
_types:
|
||||||
_notes1:
|
_notes1:
|
||||||
title: "미스키 설정하고 있었는데요"
|
title: "미스키 시작했는데요"
|
||||||
description: "첫 노트를 포스트했습니다"
|
description: "첫 노트를 작성했습니다"
|
||||||
flavor: "Misskey에 오신 것을 환영합니다!"
|
flavor: "Misskey에 오신 것을 환영합니다!"
|
||||||
_notes10:
|
_notes10:
|
||||||
title: "노트 조금"
|
title: "노트 조금"
|
||||||
@@ -962,7 +962,7 @@ _achievements:
|
|||||||
title: "노트가 어디서 솟아?"
|
title: "노트가 어디서 솟아?"
|
||||||
description: "5,000개의 노트를 작성했습니다"
|
description: "5,000개의 노트를 작성했습니다"
|
||||||
_notes10000:
|
_notes10000:
|
||||||
title: "슈퍼-노트"
|
title: "슈퍼 노트"
|
||||||
description: "10,000개의 노트를 작성했습니다"
|
description: "10,000개의 노트를 작성했습니다"
|
||||||
_notes20000:
|
_notes20000:
|
||||||
title: "노트 더 없어?"
|
title: "노트 더 없어?"
|
||||||
@@ -989,7 +989,7 @@ _achievements:
|
|||||||
title: "노트 우주"
|
title: "노트 우주"
|
||||||
description: "90,000개의 노트를 작성했습니다"
|
description: "90,000개의 노트를 작성했습니다"
|
||||||
_notes100000:
|
_notes100000:
|
||||||
title: "네 모든 노트는 내 거야"
|
title: "ALL YOUR NOTE ARE BELONG TO US"
|
||||||
description: "100,000개의 노트를 작성했습니다"
|
description: "100,000개의 노트를 작성했습니다"
|
||||||
flavor: "이만큼 쓸 일도 없겠지만... 다른 할 일이 있진 않으신가요?"
|
flavor: "이만큼 쓸 일도 없겠지만... 다른 할 일이 있진 않으신가요?"
|
||||||
_login3:
|
_login3:
|
||||||
@@ -1012,7 +1012,7 @@ _achievements:
|
|||||||
_login100:
|
_login100:
|
||||||
title: "미스키스트 III"
|
title: "미스키스트 III"
|
||||||
description: "총 100일간 로그인했습니다"
|
description: "총 100일간 로그인했습니다"
|
||||||
flavor: "그 유저, 미스키스트를 위하여"
|
flavor: "그 유저, 미스키스트이다"
|
||||||
_login200:
|
_login200:
|
||||||
title: "단골 I"
|
title: "단골 I"
|
||||||
description: "총 200일간 로그인했습니다"
|
description: "총 200일간 로그인했습니다"
|
||||||
@@ -1025,7 +1025,7 @@ _achievements:
|
|||||||
_login500:
|
_login500:
|
||||||
title: "베테랑 I"
|
title: "베테랑 I"
|
||||||
description: "총 500일간 로그인했습니다"
|
description: "총 500일간 로그인했습니다"
|
||||||
flavor: "여러분, 저 이 노트들 좋아해요"
|
flavor: "제군, 나는 노트가 좋다"
|
||||||
_login600:
|
_login600:
|
||||||
title: "베테랑 II"
|
title: "베테랑 II"
|
||||||
description: "총 600일간 로그인했습니다"
|
description: "총 600일간 로그인했습니다"
|
||||||
@@ -1041,13 +1041,16 @@ _achievements:
|
|||||||
_login1000:
|
_login1000:
|
||||||
title: "노트 마스터 III"
|
title: "노트 마스터 III"
|
||||||
description: "총 1,000일간 로그인했습니다"
|
description: "총 1,000일간 로그인했습니다"
|
||||||
flavor: "미스키를 사용해 주셔서 감사합니다!"
|
flavor: "Misskey를 사용해 주셔서 감사합니다!"
|
||||||
_noteClipped1:
|
_noteClipped1:
|
||||||
title: "클립할 수밖에 없었어"
|
title: "클립할 수밖에 없었어"
|
||||||
description: "처음으로 노트를 클립했습니다"
|
description: "처음으로 노트를 클립했습니다"
|
||||||
_noteFavorited1:
|
_noteFavorited1:
|
||||||
title: "별을 바라보는 자"
|
title: "별을 바라보는 자"
|
||||||
description: "처음으로 노트를 즐겨찾기했습니다"
|
description: "처음으로 노트를 즐겨찾기했습니다"
|
||||||
|
_myNoteFavorited1:
|
||||||
|
title: "별을 원하는 자"
|
||||||
|
description: "다른 사람이 당신의 노트를 즐겨찾기했습니다"
|
||||||
_profileFilled:
|
_profileFilled:
|
||||||
title: "준비 완료"
|
title: "준비 완료"
|
||||||
description: "프로필 설정을 완료했습니다"
|
description: "프로필 설정을 완료했습니다"
|
||||||
@@ -1074,7 +1077,7 @@ _achievements:
|
|||||||
title: "첫 팔로워"
|
title: "첫 팔로워"
|
||||||
description: "사용자가 처음으로 팔로잉했습니다"
|
description: "사용자가 처음으로 팔로잉했습니다"
|
||||||
_followers10:
|
_followers10:
|
||||||
title: "날 따라와!"
|
title: "팔로우 미!"
|
||||||
description: "10명의 사용자가 팔로우했습니다"
|
description: "10명의 사용자가 팔로우했습니다"
|
||||||
_followers50:
|
_followers50:
|
||||||
title: "이곳저곳"
|
title: "이곳저곳"
|
||||||
@@ -1092,15 +1095,18 @@ _achievements:
|
|||||||
title: "유명인사"
|
title: "유명인사"
|
||||||
description: "1,000명의 사용자가 팔로우했습니다"
|
description: "1,000명의 사용자가 팔로우했습니다"
|
||||||
_collectAchievements30:
|
_collectAchievements30:
|
||||||
title: "도전과제 콜렉터"
|
title: "도전 과제 콜렉터"
|
||||||
description: "30개의 도전과제를 획득했습니다"
|
description: "30개의 도전과제를 획득했습니다"
|
||||||
_viewAchievements3min:
|
_viewAchievements3min:
|
||||||
title: "저 도전과제 좋아해요"
|
title: "저 도전과제 좋아해요"
|
||||||
description: "도전과제 목록을 3분 이상 보세요"
|
description: "도전 과제 목록을 3분 이상 쳐다봤습니다"
|
||||||
_iLoveMisskey:
|
_iLoveMisskey:
|
||||||
title: "I Love Misskey"
|
title: "I Love Misskey"
|
||||||
description: "\"I ❤ #Misskey\"를 포스트했습니다"
|
description: "\"I ❤ #Misskey\"를 포스트했습니다"
|
||||||
flavor: "Misskey를 이용해주셔서 감사합니다! - 개발팀 일동"
|
flavor: "Misskey를 이용해주셔서 감사합니다! - 개발팀 일동"
|
||||||
|
_foundTreasure:
|
||||||
|
title: "보물찾기"
|
||||||
|
description: "숨겨진 보물을 발견했습니다"
|
||||||
_client30min:
|
_client30min:
|
||||||
title: "잠깐 쉬어"
|
title: "잠깐 쉬어"
|
||||||
description: "클라이언트를 시작하고 30분이 경과하였습니다"
|
description: "클라이언트를 시작하고 30분이 경과하였습니다"
|
||||||
@@ -1113,7 +1119,7 @@ _achievements:
|
|||||||
flavor: "잠 좀 자세요. 걱정돼요."
|
flavor: "잠 좀 자세요. 걱정돼요."
|
||||||
_postedAt0min0sec:
|
_postedAt0min0sec:
|
||||||
title: "정각"
|
title: "정각"
|
||||||
description: "1초도 어긋나지 않은 정각에 노트를 포스트했습니다"
|
description: "0분 0초 정각에 노트를 작성했습니다"
|
||||||
flavor: "째깍 째깍 째깍 땡!"
|
flavor: "째깍 째깍 째깍 땡!"
|
||||||
_selfQuote:
|
_selfQuote:
|
||||||
title: "혼잣말"
|
title: "혼잣말"
|
||||||
@@ -1121,21 +1127,24 @@ _achievements:
|
|||||||
_htl20npm:
|
_htl20npm:
|
||||||
title: "타임라인 폭주 중"
|
title: "타임라인 폭주 중"
|
||||||
description: "1분 사이에 홈 타임라인에 노트가 20개 넘게 생성되었습니다"
|
description: "1분 사이에 홈 타임라인에 노트가 20개 넘게 생성되었습니다"
|
||||||
|
_viewInstanceChart:
|
||||||
|
title: "애널리스트"
|
||||||
|
description: "인스턴스의 차트를 열었습니다"
|
||||||
_outputHelloWorldOnScratchpad:
|
_outputHelloWorldOnScratchpad:
|
||||||
title: "Hello, world!"
|
title: "Hello, world!"
|
||||||
description: "스크래치패드에서 hello world를 출력하세요"
|
description: "스크래치패드에서 hello world를 출력했습니다"
|
||||||
_open3windows:
|
_open3windows:
|
||||||
title: "멀티 윈도우"
|
title: "멀티 윈도우"
|
||||||
description: "3개 이상의 창을 여세요"
|
description: "3개 이상의 창을 열었습니다"
|
||||||
_driveFolderCircularReference:
|
_driveFolderCircularReference:
|
||||||
title: "순환 참조"
|
title: "순환 참조"
|
||||||
description: "드라이브 폴더를 자신을 가리키도록 만드려 시도했습니다"
|
description: "드라이브 폴더를 자신을 가리키도록 만드려 시도했습니다"
|
||||||
_reactWithoutRead:
|
_reactWithoutRead:
|
||||||
title: "읽고 답하긴 하시는 건가요?"
|
title: "읽고 답하긴 하시는 건가요?"
|
||||||
description: "100자가 넘는 포스트에 3초 안에 포스트했습니다"
|
description: "100자가 넘는 노트가 작성되고 3초 안에 반응했습니다"
|
||||||
_clickedClickHere:
|
_clickedClickHere:
|
||||||
title: "여길 눌러보세요"
|
title: "여길 눌러보세요"
|
||||||
description: "이 곳을 눌러봤습니다"
|
description: "여길을 눌러봤습니다"
|
||||||
_justPlainLucky:
|
_justPlainLucky:
|
||||||
title: "그냥 운이 좋았어"
|
title: "그냥 운이 좋았어"
|
||||||
description: "매 10초마다 0.01%의 확률로 달성됩니다"
|
description: "매 10초마다 0.01%의 확률로 달성됩니다"
|
||||||
@@ -1143,25 +1152,25 @@ _achievements:
|
|||||||
title: "신 콤플렉스"
|
title: "신 콤플렉스"
|
||||||
description: "이름을 syuilo로 설정했습니다"
|
description: "이름을 syuilo로 설정했습니다"
|
||||||
_passedSinceAccountCreated1:
|
_passedSinceAccountCreated1:
|
||||||
title: "1년"
|
title: "1주년"
|
||||||
description: "계정을 생성하고 1년이 지났습니다"
|
description: "계정을 생성하고 1년이 지났습니다"
|
||||||
_passedSinceAccountCreated2:
|
_passedSinceAccountCreated2:
|
||||||
title: "2년"
|
title: "2주년"
|
||||||
description: "계정을 생성하고 2년이 지났습니다"
|
description: "계정을 생성하고 2년이 지났습니다"
|
||||||
_passedSinceAccountCreated3:
|
_passedSinceAccountCreated3:
|
||||||
title: "3년"
|
title: "3주년"
|
||||||
description: "계정을 생성하고 3년이 지났습니다"
|
description: "계정을 생성하고 3년이 지났습니다"
|
||||||
_loggedInOnBirthday:
|
_loggedInOnBirthday:
|
||||||
title: "생일 축하합니다!"
|
title: "생일 축하합니다!"
|
||||||
description: "설정한 생일에 로그인했습니다"
|
description: "생일에 로그인했습니다"
|
||||||
_loggedInOnNewYearsDay:
|
_loggedInOnNewYearsDay:
|
||||||
title: "새해 복 많이 받으세요"
|
title: "새해 복 많이 받으세요"
|
||||||
description: "새해 첫 날에 로그인했습니다"
|
description: "새해 첫 날에 로그인했습니다"
|
||||||
flavor: "올해에도 저희 인스턴스에 관심을 가져 주셔서 감사합니다"
|
flavor: "올해에도 저희 인스턴스에 관심을 가져 주셔서 감사합니다"
|
||||||
_cookieClicked:
|
_cookieClicked:
|
||||||
title: "쿠키 클리커 게임"
|
title: "쿠키를 클릭하는 게임"
|
||||||
description: "쿠키를 클릭했습니다"
|
description: "쿠키를 클릭했습니다"
|
||||||
flavor: "뭔가 문제가 있나요?"
|
flavor: "소프트웨어 착각하지 않으셨나요?"
|
||||||
_brainDiver:
|
_brainDiver:
|
||||||
title: "Brain Diver"
|
title: "Brain Diver"
|
||||||
description: "Brain Diver로의 링크를 첨부했습니다"
|
description: "Brain Diver로의 링크를 첨부했습니다"
|
||||||
@@ -1857,5 +1866,6 @@ _deck:
|
|||||||
tl: "타임라인"
|
tl: "타임라인"
|
||||||
antenna: "안테나"
|
antenna: "안테나"
|
||||||
list: "리스트"
|
list: "리스트"
|
||||||
|
channel: "채널"
|
||||||
mentions: "받은 멘션"
|
mentions: "받은 멘션"
|
||||||
direct: "다이렉트"
|
direct: "다이렉트"
|
||||||
|
162
locales/lo-LA.yml
Normal file
162
locales/lo-LA.yml
Normal file
@@ -0,0 +1,162 @@
|
|||||||
|
---
|
||||||
|
_lang_: "ພາສາລາວ"
|
||||||
|
headlineMisskey: "ເຊື່ອມຕໍ່ເຄືອຂ່າຍໂດຍຫມາຍເຫດ"
|
||||||
|
introMisskey: "ຍິນດີຕ້ອນຮັບ! Misskey ເປັນແຫຼ່ງເປີດ, ການບໍລິການ microblogging ກະຈາຍ\nສ້າງ \"ບັນທຶກ\" ເພື່ອແບ່ງປັນຄວາມຄິດຂອງທ່ານກັບທຸກໆຄົນທີ່ຢູ່ອ້ອມຮອບທ່ານ 📡\nດ້ວຍ \"ປະຕິກິລິຍາ\", ທ່ານຍັງສາມາດສະແດງຄວາມຮູ້ສຶກຂອງທ່ານຢ່າງໄວວາກ່ຽວກັບບັນທຶກຂອງທຸກໆຄົນ 👍\nມາສຳຫຼວດໂລກໃໝ່! 🚀"
|
||||||
|
poweredByMisskeyDescription: "{name} ແມ່ນສ່ວນໜຶ່ງຂອງການບໍລິການທີ່ຂັບເຄື່ອນໂດຍແພລດຟອມ open source. <b>Misskey</b> (ເອີ້ນວ່າ \"Misskey instance\")"
|
||||||
|
monthAndDay: "{ເດືອນ}/{ມື້}"
|
||||||
|
search: "ຄົ້ນຫາ"
|
||||||
|
notifications: "ການແຈ້ງເຕືອນ"
|
||||||
|
username: "ຊື່ຜູ້ໃຊ້"
|
||||||
|
password: "ລະຫັດຜ່ານ"
|
||||||
|
forgotPassword: "ລືມລະຫັດຜ່ານ"
|
||||||
|
fetchingAsApObject: "ກຳລັງດຶງຂໍ້ມູນຈາກ fediverse..."
|
||||||
|
ok: "ຕົກລົງ"
|
||||||
|
gotIt: "ເຂົ້າໃຈແລ້ວ!"
|
||||||
|
cancel: "ຍົກເລີກ"
|
||||||
|
noThankYou: "ບໍ່ແມ່ນຕອນນີ້"
|
||||||
|
enterUsername: "ປ້ອນຊື່ຜູ້ໃຊ້"
|
||||||
|
renotedBy: "Renoted ໂດຍ {ຜູ້ໃຊ້}"
|
||||||
|
noNotes: "ບໍ່ມີຫມາຍເຫດ"
|
||||||
|
noNotifications: "ບໍ່ມີການແຈ້ງເຕືອນ"
|
||||||
|
instance: "ອີນສະແຕນ"
|
||||||
|
settings: "ກຳນົດຄ່າ"
|
||||||
|
basicSettings: "ການຕັ້ງຄ່າພື້ນຖານ"
|
||||||
|
otherSettings: "ການຕັ້ງຄ່າອື່ນໆ"
|
||||||
|
openInWindow: "ເປີດຢູ່ໃນປ່ອງຢ້ຽມ"
|
||||||
|
profile: "ໂພຼຟາຍ"
|
||||||
|
timeline: "ເສັ້ນກຳນົດເວລາ"
|
||||||
|
noAccountDescription: "ຜູ້ໃຊ້ນີ້ຍັງບໍ່ໄດ້ຂຽນໃນຊີວະປະຫວັດຂອງເຂົາເຈົ້າເທື່ອ"
|
||||||
|
login: "ເຂົ້າສູ່ລະບົບ"
|
||||||
|
loggingIn: "ກຳລັງເຂົ້າສູ່ລະບົບ..."
|
||||||
|
logout: "ອອກຈາກລະບົບ"
|
||||||
|
signup: "ລົງທະບຽນ"
|
||||||
|
uploading: "ການອັບໂຫຼດ..."
|
||||||
|
save: "ບັນທຶກ"
|
||||||
|
users: "ຜູ້ໃຊ້ຕ່າງໆ"
|
||||||
|
addUser: "ເພີ່ມຜູ້ໃຊ້"
|
||||||
|
favorite: "ເພີ່ມໃສ່ລາຍການທີ່ມັກ"
|
||||||
|
favorites: "ລາຍການທີ່ມັກ"
|
||||||
|
unfavorite: "ລຶບອອກຈາກລາຍການທີ່ມັກ"
|
||||||
|
favorited: "ເພີ່ມໃສ່ລາຍການທີ່ມັກແລ້ວ"
|
||||||
|
alreadyFavorited: "ເພີ່ມເຂົ້າໃນລາຍການທີ່ມັກແລ້ວ."
|
||||||
|
cantFavorite: "ບໍ່ສາມາດເພີ່ມໃສ່ລາຍການທີ່ມັກໄດ້."
|
||||||
|
pin: "ປັກໝຸດໄປຫາໂປຣໄຟລ໌"
|
||||||
|
unpin: "ຖອດປັກໝຸດອອກຈາກໂປຣໄຟລ໌"
|
||||||
|
copyContent: "ຄັດລອກເນື້ອຫາ"
|
||||||
|
copyLink: "ສຳເນົາລິ້ງ"
|
||||||
|
delete: "ລຶບ"
|
||||||
|
deleteAndEdit: "ລົບແລະແກ້ໄຂ"
|
||||||
|
deleteAndEditConfirm: "ເຈົ້າແນ່ໃຈບໍ່? ທີ່ທ່ານຕ້ອງການທີ່ຈະລຶບບັນທຶກນີ້ແລະແກ້ໄຂມັນ ທ່ານອາດຈະສູນເສຍການໂຕ້ຕອບ, ບັນທຶກ, ແລະການຕອບກັບທັງໝົດ"
|
||||||
|
addToList: "ເພີ່ມໃສ່ລາຍຊື່"
|
||||||
|
sendMessage: "ສົ່ງຂໍ້ຄວາມ"
|
||||||
|
copyRSS: "ສຳເນົາ RSS"
|
||||||
|
copyUsername: "ສຳເນົາຊື່ຜູ້ໃຊ້"
|
||||||
|
searchUser: "ຄົ້ນຫາຜູ້ໃຊ້"
|
||||||
|
reply: "ຕອບໄປທີ"
|
||||||
|
loadMore: "ໂຫຼດເພີ່ມເຕີມ"
|
||||||
|
showMore: "ໂຫຼດເພີ່ມເຕີມ"
|
||||||
|
showLess: "ປິດ"
|
||||||
|
youGotNewFollower: "ໄດ້ຕິດຕາມທ່ານ"
|
||||||
|
receiveFollowRequest: "ປະຕິບັດຕາມຄໍາຮ້ອງຂໍທີ່ໄດ້ຮັບ"
|
||||||
|
followRequestAccepted: "ຜູ້ຕິດຕາມໄດ້ຍອມຮັບຄໍາຮ້ອງຂໍຂອງທ່ານ"
|
||||||
|
mention: "ໄດ້ກ່າວມາ"
|
||||||
|
mentions: "ກ່າວເຖິງ"
|
||||||
|
directNotes: "ໂດຍກົງຫມາຍເຫດ"
|
||||||
|
importAndExport: "ນໍາເຂົ້າ / ສົ່ງອອກ"
|
||||||
|
import: "ນຳເຂົ້າ"
|
||||||
|
export: "ນຳອອກ"
|
||||||
|
files: "ໄຟລ໌"
|
||||||
|
download: "ດາວໂຫລດ"
|
||||||
|
driveFileDeleteConfirm: "ທ່ານແນ່ໃຈບໍ່ວ່າຕ້ອງການລຶບໄຟລ໌ \"{name}\"? ບັນທຶກທີ່ມີໄຟລ໌ແນບນີ້ຈະຖືກລຶບຖິ້ມ"
|
||||||
|
unfollowConfirm: "ທ່ານແນ່ໃຈບໍ່ວ່າຕ້ອງການເຊົາຕິດຕາມ {name}?"
|
||||||
|
exportRequested: "ໃນເວລາທີ່ທ່ານໄດ້ຮ້ອງຂໍການສົ່ງອອກ ມັນອາດຈະໃຊ້ເວລາບາງເວລາ ແລະມັນຈະຖືກເພີ່ມໃສ່ drive ຂອງທ່ານເມື່ອມັນສຳເລັດແລ້ວ"
|
||||||
|
importRequested: "ໃນເວລາທີ່ທ່ານໄດ້ຮ້ອງຂໍການນໍາເຂົ້າ ມັນອາດຈະໃຊ້ເວລາບາງເວລາ"
|
||||||
|
lists: "ລາຍການ"
|
||||||
|
noLists: "ທ່ານບໍ່ມີລາຍການໃດໆ"
|
||||||
|
note: "ບັນທຶກ"
|
||||||
|
notes: "ບັນທຶກ"
|
||||||
|
following: "ກຳລັງຕິດຕາມ"
|
||||||
|
followers: "ຜູ້ຕິດຕາມ"
|
||||||
|
followsYou: "ຕິດຕາມເຈົ້າ"
|
||||||
|
createList: "ສ້າງລາຍຊື່"
|
||||||
|
manageLists: "ການບໍລິຫານບັນຊີລາຍການ"
|
||||||
|
error: "ຂໍ້ຜິດພາດ"
|
||||||
|
somethingHappened: "ອຸຍ, ມີບາງຢ່າງຜິດພາດ"
|
||||||
|
retry: "ລອງໃຫມ່"
|
||||||
|
pageLoadError: "ເກີດຄວາມຜິດພາດໃນການໂຫລດໜ້ານີ້"
|
||||||
|
pageLoadErrorDescription: "ປົກກະຕິແລ້ວມັນເກີດຈາກຄວາມຜິດພາດເຄືອຂ່າຍ ຫຼື cache ຂອງຕົວທ່ອງເວັບ ລອງລຶບລ້າງແຄດແລ້ວລອງໃໝ່ພາຍຫຼັງສອງສາມນາທີ"
|
||||||
|
serverIsDead: "ເຊີບເວີນີ້ບໍ່ຕອບສະໜອງ ກະລຸນາລໍຖ້າຈັກໜ່ອຍແລ້ວລອງໃໝ່ອີກຄັ້ງ"
|
||||||
|
youShouldUpgradeClient: "ເພື່ອເບິ່ງໜ້ານີ້, ກະລຸນາໂຫຼດຂໍ້ມູນຄືນໃໝ່ເພື່ອອັບເດດລູກຄ້າຂອງທ່ານ"
|
||||||
|
enterListName: "ໃສ່ຊື່ສຳລັບລາຍຊື່"
|
||||||
|
privacy: "ຄວາມເປັນສ່ວນຕົວ"
|
||||||
|
makeFollowManuallyApprove: "ປະຕິບັດຕາມການຮ້ອງຂໍຮຽກຮ້ອງໃຫ້ມີການອະນຸມັດ"
|
||||||
|
defaultNoteVisibility: "ເປັນຄ່າເລີ່ມຕົ້ນ"
|
||||||
|
follow: "ກຳລັງຕິດຕາມ"
|
||||||
|
followRequest: "ສົ່ງການຮ້ອງຂໍປະຕິບຕາມ"
|
||||||
|
followRequests: "ປະຕິບັດຕາມຄໍາຮ້ອງຂໍ"
|
||||||
|
unfollow: "ເຊົາຕິດຕາມ"
|
||||||
|
followRequestPending: "ປະຕິບັດຕາມຄໍາຮ້ອງຂໍທີ່ລໍຖ້າຢູ່"
|
||||||
|
enterEmoji: "ປ້ອນອີໂມຈິ"
|
||||||
|
renote: "Renote"
|
||||||
|
unrenote: "ເລີກ Renote"
|
||||||
|
pinned: "ປັກໝຸດໄປຫາໂປຣໄຟລ໌"
|
||||||
|
addAccount: "ເພີ່ມບັນຊີ"
|
||||||
|
loginFailed: "ການເຂົ້າສູ່ລະບົບບໍ່ສຳເລັດ"
|
||||||
|
general: "ທົ່ວໄປ"
|
||||||
|
wallpaper: "ພາບພື້ນຫລັງ"
|
||||||
|
setWallpaper: "ຕັ້ງເປັນພາບພື້ນຫຼັງ"
|
||||||
|
instances: "ອີນສະແຕນ"
|
||||||
|
statistics: "ສະຖິຕິ"
|
||||||
|
clearQueue: "ລ້າງຄິວ"
|
||||||
|
clearCachedFiles: "ລຶບລ້າງແຄສ"
|
||||||
|
editProfile: "ແກ້ໄຂໂປຣໄຟລ໌"
|
||||||
|
remove: "ລຶບ"
|
||||||
|
userList: "ລາຍການ"
|
||||||
|
smtpUser: "ຊື່ຜູ້ໃຊ້"
|
||||||
|
smtpPass: "ລະຫັດຜ່ານ"
|
||||||
|
clearCache: "ລຶບລ້າງແຄສ"
|
||||||
|
user: "ຜູ້ໃຊ້ຕ່າງໆ"
|
||||||
|
searchByGoogle: "ຄົ້ນຫາ"
|
||||||
|
file: "ໄຟລ໌"
|
||||||
|
_email:
|
||||||
|
_follow:
|
||||||
|
title: "ໄດ້ຕິດຕາມທ່ານ"
|
||||||
|
_mfm:
|
||||||
|
mention: "ໄດ້ກ່າວມາ"
|
||||||
|
search: "ຄົ້ນຫາ"
|
||||||
|
_theme:
|
||||||
|
keys:
|
||||||
|
mention: "ໄດ້ກ່າວມາ"
|
||||||
|
renote: "Renote"
|
||||||
|
_sfx:
|
||||||
|
note: "ບັນທຶກ"
|
||||||
|
notification: "ການແຈ້ງເຕືອນ"
|
||||||
|
_widgets:
|
||||||
|
profile: "ໂພຼຟາຍ"
|
||||||
|
notifications: "ການແຈ້ງເຕືອນ"
|
||||||
|
timeline: "ເສັ້ນກຳນົດເວລາ"
|
||||||
|
_cw:
|
||||||
|
show: "ໂຫຼດເພີ່ມເຕີມ"
|
||||||
|
_visibility:
|
||||||
|
followers: "ຜູ້ຕິດຕາມ"
|
||||||
|
_profile:
|
||||||
|
username: "ຊື່ຜູ້ໃຊ້"
|
||||||
|
_exportOrImport:
|
||||||
|
followingList: "ກຳລັງຕິດຕາມ"
|
||||||
|
userLists: "ລາຍການ"
|
||||||
|
_notification:
|
||||||
|
youWereFollowed: "ໄດ້ຕິດຕາມທ່ານ"
|
||||||
|
_types:
|
||||||
|
follow: "ກຳລັງຕິດຕາມ"
|
||||||
|
mention: "ໄດ້ກ່າວມາ"
|
||||||
|
renote: "Renote"
|
||||||
|
_actions:
|
||||||
|
reply: "ຕອບໄປທີ"
|
||||||
|
renote: "Renote"
|
||||||
|
_deck:
|
||||||
|
_columns:
|
||||||
|
notifications: "ການແຈ້ງເຕືອນ"
|
||||||
|
tl: "ເສັ້ນກຳນົດເວລາ"
|
||||||
|
list: "ລາຍການ"
|
||||||
|
channel: "ຊ່ອງ"
|
||||||
|
mentions: "ກ່າວເຖິງ"
|
@@ -1438,5 +1438,6 @@ _deck:
|
|||||||
tl: "Oś czasu"
|
tl: "Oś czasu"
|
||||||
antenna: "Anteny"
|
antenna: "Anteny"
|
||||||
list: "Listy"
|
list: "Listy"
|
||||||
|
channel: "Kanały"
|
||||||
mentions: "Wspomnienia"
|
mentions: "Wspomnienia"
|
||||||
direct: "Bezpośredni"
|
direct: "Bezpośredni"
|
||||||
|
@@ -721,4 +721,5 @@ _deck:
|
|||||||
tl: "Cronologie"
|
tl: "Cronologie"
|
||||||
antenna: "Antene"
|
antenna: "Antene"
|
||||||
list: "Liste"
|
list: "Liste"
|
||||||
|
channel: "Canale"
|
||||||
mentions: "Mențiuni"
|
mentions: "Mențiuni"
|
||||||
|
@@ -22,7 +22,7 @@ instance: "Инстанс"
|
|||||||
settings: "Настройки"
|
settings: "Настройки"
|
||||||
basicSettings: "Основные настройки"
|
basicSettings: "Основные настройки"
|
||||||
otherSettings: "Прочие настройки"
|
otherSettings: "Прочие настройки"
|
||||||
openInWindow: "Открывать в плавающих окнах"
|
openInWindow: "Открыть в плавающем окне"
|
||||||
profile: "Профиль"
|
profile: "Профиль"
|
||||||
timeline: "Лента"
|
timeline: "Лента"
|
||||||
noAccountDescription: "Пользователь ничего не написал про себя"
|
noAccountDescription: "Пользователь ничего не написал про себя"
|
||||||
@@ -273,7 +273,7 @@ light: "Светлый"
|
|||||||
dark: "Тёмный"
|
dark: "Тёмный"
|
||||||
lightThemes: "Светлые темы"
|
lightThemes: "Светлые темы"
|
||||||
darkThemes: "Тёмные темы"
|
darkThemes: "Тёмные темы"
|
||||||
syncDeviceDarkMode: "Синхронизировать с темным режимом устройства"
|
syncDeviceDarkMode: "Синхронизировать с тёмной темой системы"
|
||||||
drive: "Диск"
|
drive: "Диск"
|
||||||
fileName: "Имя файла"
|
fileName: "Имя файла"
|
||||||
selectFile: "Выберите файл"
|
selectFile: "Выберите файл"
|
||||||
@@ -456,6 +456,7 @@ uiLanguage: "Язык интерфейса"
|
|||||||
groupInvited: "Приглашение в группу"
|
groupInvited: "Приглашение в группу"
|
||||||
aboutX: "Описание {x}"
|
aboutX: "Описание {x}"
|
||||||
emojiStyle: "Стиль эмодзи"
|
emojiStyle: "Стиль эмодзи"
|
||||||
|
native: "Системные"
|
||||||
disableDrawer: "Не использовать выдвижные меню"
|
disableDrawer: "Не использовать выдвижные меню"
|
||||||
youHaveNoGroups: "У вас нет ни одной группы"
|
youHaveNoGroups: "У вас нет ни одной группы"
|
||||||
joinOrCreateGroup: "Получайте приглашения в группы или создавайте свои собственные"
|
joinOrCreateGroup: "Получайте приглашения в группы или создавайте свои собственные"
|
||||||
@@ -603,6 +604,7 @@ smtpSecureInfo: "Выключите при использовании STARTTLS."
|
|||||||
testEmail: "Проверка доставки электронной почты"
|
testEmail: "Проверка доставки электронной почты"
|
||||||
wordMute: "Скрытие слов"
|
wordMute: "Скрытие слов"
|
||||||
regexpError: "Ошибка в регулярном выражении"
|
regexpError: "Ошибка в регулярном выражении"
|
||||||
|
regexpErrorDescription: "В списке {tab} скрытых слов, в строке {line} обнаружена синтаксическая ошибка:"
|
||||||
instanceMute: "Глушение инстансов"
|
instanceMute: "Глушение инстансов"
|
||||||
userSaysSomething: "{name} что-то сообщает"
|
userSaysSomething: "{name} что-то сообщает"
|
||||||
makeActive: "Активировать"
|
makeActive: "Активировать"
|
||||||
@@ -804,7 +806,7 @@ translate: "Перевод"
|
|||||||
translatedFrom: "Перевод. Язык оригинала — {x}"
|
translatedFrom: "Перевод. Язык оригинала — {x}"
|
||||||
accountDeletionInProgress: "В настоящее время выполняется удаление учетной записи"
|
accountDeletionInProgress: "В настоящее время выполняется удаление учетной записи"
|
||||||
usernameInfo: "Имя, которое отличает вашу учетную запись от других на этом сервере. Вы можете использовать алфавит (a~z, A~Z), цифры (0~9) или символы подчеркивания (_). Имена пользователей не могут быть изменены позже."
|
usernameInfo: "Имя, которое отличает вашу учетную запись от других на этом сервере. Вы можете использовать алфавит (a~z, A~Z), цифры (0~9) или символы подчеркивания (_). Имена пользователей не могут быть изменены позже."
|
||||||
aiChanMode: "ИИ режим"
|
aiChanMode: "Режим Ай"
|
||||||
keepCw: "Сохраняйте Предупреждения о содержимом"
|
keepCw: "Сохраняйте Предупреждения о содержимом"
|
||||||
pubSub: "Учётные записи Pub/Sub"
|
pubSub: "Учётные записи Pub/Sub"
|
||||||
lastCommunication: "Последнее сообщение"
|
lastCommunication: "Последнее сообщение"
|
||||||
@@ -821,8 +823,8 @@ manageAccounts: "Управление аккаунтом"
|
|||||||
makeReactionsPublic: "Опубликовать список реакций"
|
makeReactionsPublic: "Опубликовать список реакций"
|
||||||
makeReactionsPublicDescription: "Список сделанных вами реакций доступен для просмотра всем желающим."
|
makeReactionsPublicDescription: "Список сделанных вами реакций доступен для просмотра всем желающим."
|
||||||
classic: "Классика"
|
classic: "Классика"
|
||||||
muteThread: "Заглушить цепочку"
|
muteThread: "Скрыть цепочку"
|
||||||
unmuteThread: "Отменить глушение цепочки"
|
unmuteThread: "Отменить сокрытие цепочки"
|
||||||
ffVisibility: "Видимость подписок и подписчиков"
|
ffVisibility: "Видимость подписок и подписчиков"
|
||||||
ffVisibilityDescription: "Здесь можно настроить, кто будет видеть ваши подписки и подписчиков."
|
ffVisibilityDescription: "Здесь можно настроить, кто будет видеть ваши подписки и подписчиков."
|
||||||
continueThread: "Показать следующие ответы"
|
continueThread: "Показать следующие ответы"
|
||||||
@@ -891,19 +893,49 @@ cannotUploadBecauseNoFreeSpace: "Файл не может быть загруж
|
|||||||
beta: "Бета"
|
beta: "Бета"
|
||||||
enableAutoSensitive: "Автоматическое определение NSFW"
|
enableAutoSensitive: "Автоматическое определение NSFW"
|
||||||
enableAutoSensitiveDescription: "Если доступно, используйте машинное обучение для автоматической установки флага NSFW на носителе. Даже если эта функция отключена, она может быть установлена автоматически в зависимости от инстанта."
|
enableAutoSensitiveDescription: "Если доступно, используйте машинное обучение для автоматической установки флага NSFW на носителе. Даже если эта функция отключена, она может быть установлена автоматически в зависимости от инстанта."
|
||||||
|
activeEmailValidationDescription: "Если включено, будет проводиться более строгая проверка адреса электронной почты, в том числе на то, что он действительный и не временный. Если же отключено, то проверяется только корректность написания адреса."
|
||||||
|
navbar: "Панель навигации"
|
||||||
|
shuffle: "Перемешать"
|
||||||
account: "Учётные записи"
|
account: "Учётные записи"
|
||||||
|
move: "Переместить"
|
||||||
|
pushNotification: "Push-уведомления"
|
||||||
|
subscribePushNotification: "Включить push-уведомления"
|
||||||
|
unsubscribePushNotification: "Выключить push-уведомления"
|
||||||
|
pushNotificationAlreadySubscribed: "Push-уведомления уже включены"
|
||||||
|
pushNotificationNotSupported: "Push-уведмления не поддерживаются инстансом или браузером"
|
||||||
|
sendPushNotificationReadMessage: "Удалять push-уведомления когда сообщение или прочитано"
|
||||||
|
sendPushNotificationReadMessageCaption: "На мгновение появится уведомление \"{emptyPushNotificationMessage}\". Расход заряда батареи может увеличиться "
|
||||||
windowMaximize: "Развернуть"
|
windowMaximize: "Развернуть"
|
||||||
windowRestore: "Восстановить"
|
windowRestore: "Восстановить"
|
||||||
|
caption: "Подпись (Automatic Translation)"
|
||||||
loggedInAsBot: "Вы под аккаунтом бота!"
|
loggedInAsBot: "Вы под аккаунтом бота!"
|
||||||
|
tools: "Инструменты"
|
||||||
|
cannotLoad: "Не удалось загрузить"
|
||||||
|
numberOfProfileView: "Количество профилей для просмотра"
|
||||||
like: "Нравится!"
|
like: "Нравится!"
|
||||||
unlike: "Отменить «нравится»"
|
unlike: "Отменить «нравится»"
|
||||||
|
numberOfLikes: "Количество лайков"
|
||||||
show: "Отображение"
|
show: "Отображение"
|
||||||
|
neverShow: "Больше не показывать"
|
||||||
|
remindMeLater: "Напомнить позже"
|
||||||
|
didYouLikeMisskey: "Вам нравится Misskey?"
|
||||||
pleaseDonate: "Сайт {host} работает на Misskey. Это бесплатное программное обеспечение, и ваши пожертвования очень бы помогли продолжать его разработку!"
|
pleaseDonate: "Сайт {host} работает на Misskey. Это бесплатное программное обеспечение, и ваши пожертвования очень бы помогли продолжать его разработку!"
|
||||||
roles: "Роли"
|
roles: "Роли"
|
||||||
role: "Роль"
|
role: "Роль"
|
||||||
|
normalUser: "Обычный пользователь"
|
||||||
|
undefined: "неопределён"
|
||||||
|
assign: "Назначить"
|
||||||
|
unassign: "Отменить назначение"
|
||||||
color: "Цвет"
|
color: "Цвет"
|
||||||
|
manageCustomEmojis: "Управлять пользовательскими эмодзи"
|
||||||
|
youCannotCreateAnymore: "Вы достигли лимита создания."
|
||||||
|
cannotPerformTemporary: "Временно недоступен"
|
||||||
|
cannotPerformTemporaryDescription: "Это действие временно невозможно выполнить из-за превышения лимита выполнения."
|
||||||
|
preset: "Шаблоны"
|
||||||
|
selectFromPresets: "Выбрать из шаблонов"
|
||||||
achievements: "Достижения"
|
achievements: "Достижения"
|
||||||
_achievements:
|
_achievements:
|
||||||
|
earnedAt: "Разблокировано в"
|
||||||
_types:
|
_types:
|
||||||
_notes1:
|
_notes1:
|
||||||
title: "Первые шаги в Misskey"
|
title: "Первые шаги в Misskey"
|
||||||
@@ -975,6 +1007,7 @@ _achievements:
|
|||||||
_login100:
|
_login100:
|
||||||
title: "Мискиец Ⅲ"
|
title: "Мискиец Ⅲ"
|
||||||
description: "100 дней на сайте"
|
description: "100 дней на сайте"
|
||||||
|
flavor: "Жестокий мискиец"
|
||||||
_login200:
|
_login200:
|
||||||
title: "Завсегдатай Ⅰ"
|
title: "Завсегдатай Ⅰ"
|
||||||
description: "200 дней на сайте"
|
description: "200 дней на сайте"
|
||||||
@@ -1010,6 +1043,9 @@ _achievements:
|
|||||||
_noteFavorited1:
|
_noteFavorited1:
|
||||||
title: "Смотрящий на звёзды"
|
title: "Смотрящий на звёзды"
|
||||||
description: "Первое добавление в избранное"
|
description: "Первое добавление в избранное"
|
||||||
|
_myNoteFavorited1:
|
||||||
|
title: "В поиске звёзд"
|
||||||
|
description: "Кому-то понравилась ваша заметка"
|
||||||
_profileFilled:
|
_profileFilled:
|
||||||
title: "Приготовления закончены"
|
title: "Приготовления закончены"
|
||||||
description: "Заполнен профиль"
|
description: "Заполнен профиль"
|
||||||
@@ -1063,6 +1099,9 @@ _achievements:
|
|||||||
title: "Я люблю Misskey"
|
title: "Я люблю Misskey"
|
||||||
description: "Написана заметка «I ❤ #Misskey»"
|
description: "Написана заметка «I ❤ #Misskey»"
|
||||||
flavor: "Спасибо за поддержку Misskey! Ваша команда разработчиков"
|
flavor: "Спасибо за поддержку Misskey! Ваша команда разработчиков"
|
||||||
|
_foundTreasure:
|
||||||
|
title: "Охота за сокровищами"
|
||||||
|
description: "Найдено спрятанное сокровище"
|
||||||
_client30min:
|
_client30min:
|
||||||
title: "Перерыв на обед"
|
title: "Перерыв на обед"
|
||||||
description: "Прошло 30 минут с момента запуска клиента"
|
description: "Прошло 30 минут с момента запуска клиента"
|
||||||
@@ -1083,6 +1122,9 @@ _achievements:
|
|||||||
_htl20npm:
|
_htl20npm:
|
||||||
title: "В потоке"
|
title: "В потоке"
|
||||||
description: "Достигнута скорость домашней ленты в 20 з/мин (заметок минуту)"
|
description: "Достигнута скорость домашней ленты в 20 з/мин (заметок минуту)"
|
||||||
|
_viewInstanceChart:
|
||||||
|
title: "Аналитик"
|
||||||
|
description: "Просмотрены статистические диаграммы инстанса"
|
||||||
_outputHelloWorldOnScratchpad:
|
_outputHelloWorldOnScratchpad:
|
||||||
title: "Привет, мир!"
|
title: "Привет, мир!"
|
||||||
description: "Выведен текст «hello world» в Когтеточке"
|
description: "Выведен текст «hello world» в Когтеточке"
|
||||||
@@ -1130,14 +1172,24 @@ _achievements:
|
|||||||
flavor: "Мисски-Мисски Ла-Ту-Ма"
|
flavor: "Мисски-Мисски Ла-Ту-Ма"
|
||||||
_role:
|
_role:
|
||||||
new: "Новая роль"
|
new: "Новая роль"
|
||||||
|
edit: "Изменить роль"
|
||||||
name: "Название роли"
|
name: "Название роли"
|
||||||
description: "Описание роли"
|
description: "Описание роли"
|
||||||
permission: "Ролевые полномочия"
|
permission: "Ролевые полномочия"
|
||||||
|
descriptionOfPermission: "<b>Модераторы</b> могут изменять базовые операции для модераторов.\n<b>Администраторы</b> могут изменять полностью настройки инстанса."
|
||||||
assignTarget: "Метод присвоения"
|
assignTarget: "Метод присвоения"
|
||||||
|
descriptionOfAssignTarget: "<b>Вручную</b> чтобы указать кому выдавать роль, а кому нет.\n<b>По условию<b> чтобы автоматически выдавать и удалять роль при условиях."
|
||||||
manual: "Вручную"
|
manual: "Вручную"
|
||||||
conditional: "По условию"
|
conditional: "По условию"
|
||||||
|
condition: "Условия"
|
||||||
|
isConditionalRole: "Эта роль выдаётся по условию."
|
||||||
isPublic: "Общедоступная роль"
|
isPublic: "Общедоступная роль"
|
||||||
descriptionOfIsPublic: "Список тех, кому назначена эта роль будет доступен всем. Кроме того эта роль будет отмечена у каждого в профиле."
|
descriptionOfIsPublic: "Список тех, кому назначена эта роль будет доступен всем. Кроме того эта роль будет отмечена у каждого в профиле."
|
||||||
|
options: "Настройки ролей"
|
||||||
|
policies: "Политики"
|
||||||
|
baseRole: "Шаблон роли"
|
||||||
|
useBaseValue: "Использовать значение из шаблона"
|
||||||
|
chooseRoleToAssign: "Выберите роль, которую хотите выдать"
|
||||||
canEditMembersByModerator: "Могут назначать модераторы"
|
canEditMembersByModerator: "Могут назначать модераторы"
|
||||||
descriptionOfCanEditMembersByModerator: "Если включено, на эту роль могут назначать пользователей как администраторы, так и модераторы. Если выключено, назначать могут только администраторы."
|
descriptionOfCanEditMembersByModerator: "Если включено, на эту роль могут назначать пользователей как администраторы, так и модераторы. Если выключено, назначать могут только администраторы."
|
||||||
priority: "Приоритет"
|
priority: "Приоритет"
|
||||||
@@ -1145,6 +1197,35 @@ _role:
|
|||||||
low: "Низкий"
|
low: "Низкий"
|
||||||
middle: "Средне"
|
middle: "Средне"
|
||||||
high: "Высокий"
|
high: "Высокий"
|
||||||
|
_options:
|
||||||
|
gtlAvailable: "Может просматривать глобальную ленту"
|
||||||
|
ltlAvailable: "Может просматривать местную ленту"
|
||||||
|
canPublicNote: "Может публиковать общедоступные заметки"
|
||||||
|
canInvite: "Может создавать пригласительные коды"
|
||||||
|
canManageCustomEmojis: "Управлять пользовательскими эмодзи"
|
||||||
|
driveCapacity: "Доступное пространство на «диске»"
|
||||||
|
pinMax: "Доступное количество закреплённых заметок"
|
||||||
|
antennaMax: "Доступное количество антенн"
|
||||||
|
wordMuteMax: "Доступное количество знаков в списке скрытия слов"
|
||||||
|
clipMax: "Максимальное количество подборок"
|
||||||
|
noteEachClipsMax: "Максимальное количество заметок в подборке"
|
||||||
|
userListMax: "Максимальное количество списков аккаунтов"
|
||||||
|
userEachUserListsMax: "Максимальное количество аккаунтов в списке"
|
||||||
|
rateLimitFactor: "Ограничение активности"
|
||||||
|
descriptionOfRateLimitFactor: "Меньшее значение — слабые ограничения, большее — сильные"
|
||||||
|
canHideAds: "Может скрыть рекламу"
|
||||||
|
_condition:
|
||||||
|
isLocal: "Местный"
|
||||||
|
isRemote: "Неместный"
|
||||||
|
createdLessThan: "Аккаунт младше, чем..."
|
||||||
|
createdMoreThan: "Аккаунт старше, чем..."
|
||||||
|
followersLessThanOrEq: "Количество подписчиков не превышает…"
|
||||||
|
followersMoreThanOrEq: "Количество подписчиков не меньше чем…"
|
||||||
|
followingLessThanOrEq: "Количество подписок не превышает…"
|
||||||
|
followingMoreThanOrEq: "Количество подписок не меньше чем…"
|
||||||
|
and: "Выполнено несколько условий:.."
|
||||||
|
or: "Выполнено любое из условий:.."
|
||||||
|
not: "Кроме тех, у кого…"
|
||||||
_sensitiveMediaDetection:
|
_sensitiveMediaDetection:
|
||||||
description: "Машинное обучение может быть использовано для автоматического обнаружения чувствительных медиа для модерации. Нагрузка на сервер увеличивается незначительно."
|
description: "Машинное обучение может быть использовано для автоматического обнаружения чувствительных медиа для модерации. Нагрузка на сервер увеличивается незначительно."
|
||||||
setSensitiveFlagAutomatically: "Установить флаг NSFW"
|
setSensitiveFlagAutomatically: "Установить флаг NSFW"
|
||||||
@@ -1191,6 +1272,24 @@ _plugin:
|
|||||||
install: "Установка расширений"
|
install: "Установка расширений"
|
||||||
installWarn: "Пожалуйста, не устанавливайте расширения, которым не доверяете."
|
installWarn: "Пожалуйста, не устанавливайте расширения, которым не доверяете."
|
||||||
manage: "Управление расширениями"
|
manage: "Управление расширениями"
|
||||||
|
_preferencesBackups:
|
||||||
|
list: "Существующие резервные копии"
|
||||||
|
saveNew: "Создать резервную копию"
|
||||||
|
loadFile: "Прочесть из файла"
|
||||||
|
apply: "Восстановить на это устройство"
|
||||||
|
save: "Обновить из текущих настроек"
|
||||||
|
inputName: "Введите название для резервной копии"
|
||||||
|
cannotSave: "Сохранить не удалось"
|
||||||
|
nameAlreadyExists: "Резервная копия под названием «{name}» уже существует. Придумайте другое."
|
||||||
|
applyConfirm: "Правда хотите загрузить резервную копию «{name}» на это устройство? Этим будут потеряны текущие настройки."
|
||||||
|
saveConfirm: "Сохранить резервную копию под названием «{name}»?"
|
||||||
|
deleteConfirm: "Удалить резервную копию «{name}»?"
|
||||||
|
renameConfirm: "Переименовать резервную копию «{old}» в «{new}»?"
|
||||||
|
noBackups: "Здесь ещё нет резервных копий. Вы можете создать резервную копию настроек на этом сайте с помощью кнопки «Создать резервную копию»."
|
||||||
|
createdAt: "Создана {date} в {time}"
|
||||||
|
updatedAt: "Обновлена {date} в {time}"
|
||||||
|
cannotLoad: "Загрузить не удалось"
|
||||||
|
invalidFile: "Некорректный формат файла"
|
||||||
_registry:
|
_registry:
|
||||||
scope: "Область"
|
scope: "Область"
|
||||||
key: "Ключ"
|
key: "Ключ"
|
||||||
@@ -1274,6 +1373,8 @@ _mfm:
|
|||||||
sparkleDescription: "Добавляет эффект искрящихся частиц."
|
sparkleDescription: "Добавляет эффект искрящихся частиц."
|
||||||
rotate: "Повернуть"
|
rotate: "Повернуть"
|
||||||
rotateDescription: "Поворачивает на заданный угол."
|
rotateDescription: "Поворачивает на заданный угол."
|
||||||
|
plain: "Буквально"
|
||||||
|
plainDescription: "MFM внутри отключается, и текст отображается как есть"
|
||||||
_instanceTicker:
|
_instanceTicker:
|
||||||
none: "Не показывать"
|
none: "Не показывать"
|
||||||
remote: "Только для других сайтов"
|
remote: "Только для других сайтов"
|
||||||
@@ -1303,12 +1404,14 @@ _wordMute:
|
|||||||
muteWordsDescription2: "Здесь можно использовать регулярные выражения — просто заключите их между двумя дробными чертами (/)."
|
muteWordsDescription2: "Здесь можно использовать регулярные выражения — просто заключите их между двумя дробными чертами (/)."
|
||||||
softDescription: "Соответствующие условиям заметки будут спрятаны из вашей ленты."
|
softDescription: "Соответствующие условиям заметки будут спрятаны из вашей ленты."
|
||||||
hardDescription: "Соответстующие условиям заметки вообще не будут попадать в вашу ленту. Даже если вы поменяете условия, отсеенные таким образом заметки уже не появятся."
|
hardDescription: "Соответстующие условиям заметки вообще не будут попадать в вашу ленту. Даже если вы поменяете условия, отсеенные таким образом заметки уже не появятся."
|
||||||
soft: "Мягкий"
|
soft: "Мягко"
|
||||||
hard: "Жёсткий"
|
hard: "Жёстко"
|
||||||
mutedNotes: "Скрытые заметки"
|
mutedNotes: "Скрытые заметки"
|
||||||
_instanceMute:
|
_instanceMute:
|
||||||
|
instanceMuteDescription: "Заметки и репосты с указанных здесь инстансов, а также ответы пользователям оттуда же не будут отображаться."
|
||||||
|
instanceMuteDescription2: "Пишите каждый инстанс на отдельной строке"
|
||||||
title: "Скрывает заметки с заданных инстансов."
|
title: "Скрывает заметки с заданных инстансов."
|
||||||
heading: "Список заглушенных инстансов"
|
heading: "Список скрытых инстансов"
|
||||||
_theme:
|
_theme:
|
||||||
explore: "Обзор"
|
explore: "Обзор"
|
||||||
install: "Установить тему"
|
install: "Установить тему"
|
||||||
@@ -1429,12 +1532,16 @@ _tutorial:
|
|||||||
step7_1: "На этом вводный урок по использованию Misskey закончен. Спасибо, что прошли его до конца!"
|
step7_1: "На этом вводный урок по использованию Misskey закончен. Спасибо, что прошли его до конца!"
|
||||||
step7_2: "Хотите изучить Misskey глубже — добро пожаловать в раздел «{help}»."
|
step7_2: "Хотите изучить Misskey глубже — добро пожаловать в раздел «{help}»."
|
||||||
step7_3: "Приятно вам провести время с Misskey🚀"
|
step7_3: "Приятно вам провести время с Misskey🚀"
|
||||||
|
step8_1: "Ах, да, не хотите ли включить push-уведомления?"
|
||||||
|
step8_2: "С push-уведомлениями вы будете в курсе репостов, ответов, реакций и всего такого, даже когда закрыли Misskey."
|
||||||
|
step8_3: "Эту настройку вы всегда сможете поменять"
|
||||||
_2fa:
|
_2fa:
|
||||||
alreadyRegistered: "Двухфакторная аутентификация уже настроена."
|
alreadyRegistered: "Двухфакторная аутентификация уже настроена."
|
||||||
registerDevice: "Зарегистрируйте ваше устройство"
|
registerDevice: "Зарегистрируйте ваше устройство"
|
||||||
registerKey: "Зарегистрировать ключ"
|
registerKey: "Зарегистрировать ключ"
|
||||||
step1: "Прежде всего, установите на устройство приложение для аутентификации, например, {a} или {b}."
|
step1: "Прежде всего, установите на устройство приложение для аутентификации, например, {a} или {b}."
|
||||||
step2: "Далее отсканируйте отображаемый QR-код при помощи приложения."
|
step2: "Далее отсканируйте отображаемый QR-код при помощи приложения."
|
||||||
|
step2Url: "Если пользуетесь приложением на компьютере, можете ввести в него эту строку (URL):"
|
||||||
step3: "И наконец, введите код, который покажет приложение."
|
step3: "И наконец, введите код, который покажет приложение."
|
||||||
step4: "Теперь при каждом входе на сайт вам нужно будет вводить код из приложения аналогичным образом."
|
step4: "Теперь при каждом входе на сайт вам нужно будет вводить код из приложения аналогичным образом."
|
||||||
securityKeyInfo: "Вы можете настроить вход с помощью аппаратного ключа безопасности, поддерживающего FIDO2, или отпечатка пальца или PIN-кода на устройстве."
|
securityKeyInfo: "Вы можете настроить вход с помощью аппаратного ключа безопасности, поддерживающего FIDO2, или отпечатка пальца или PIN-кода на устройстве."
|
||||||
@@ -1451,7 +1558,7 @@ _permissions:
|
|||||||
"write:following": "Изменять спискок подписок"
|
"write:following": "Изменять спискок подписок"
|
||||||
"read:messaging": "Смотреть сообщения"
|
"read:messaging": "Смотреть сообщения"
|
||||||
"write:messaging": "Писать и удалять сообщения"
|
"write:messaging": "Писать и удалять сообщения"
|
||||||
"read:mutes": "Смотреть спискок скрытых пользователей"
|
"read:mutes": "Смотреть список скрытых пользователей"
|
||||||
"write:mutes": "Изменять список скрытых пользователей"
|
"write:mutes": "Изменять список скрытых пользователей"
|
||||||
"write:notes": "Писать и удалять заметки"
|
"write:notes": "Писать и удалять заметки"
|
||||||
"read:notifications": "Смотреть уведомления"
|
"read:notifications": "Смотреть уведомления"
|
||||||
@@ -1502,10 +1609,13 @@ _widgets:
|
|||||||
trends: "Актуальное"
|
trends: "Актуальное"
|
||||||
clock: "Часы"
|
clock: "Часы"
|
||||||
rss: "Просмотр RSS"
|
rss: "Просмотр RSS"
|
||||||
|
rssTicker: "Бегущая строка RSS"
|
||||||
activity: "Активность"
|
activity: "Активность"
|
||||||
photos: "Фото"
|
photos: "Фото"
|
||||||
digitalClock: "Цифровые часы"
|
digitalClock: "Цифровые часы"
|
||||||
|
unixClock: "Часы UNIX"
|
||||||
federation: "Федерация"
|
federation: "Федерация"
|
||||||
|
instanceCloud: "Облако инстансов"
|
||||||
postForm: "Форма отправки"
|
postForm: "Форма отправки"
|
||||||
slideshow: "Показ слайдов"
|
slideshow: "Показ слайдов"
|
||||||
button: "Кнопка"
|
button: "Кнопка"
|
||||||
@@ -1513,9 +1623,12 @@ _widgets:
|
|||||||
jobQueue: "Очередь заданий"
|
jobQueue: "Очередь заданий"
|
||||||
serverMetric: "Показатели сервера"
|
serverMetric: "Показатели сервера"
|
||||||
aiscript: "Консоль AiScript"
|
aiscript: "Консоль AiScript"
|
||||||
|
aiscriptApp: "Приложение на AiScript"
|
||||||
aichan: "Ай"
|
aichan: "Ай"
|
||||||
|
userList: "Список аккаунтов"
|
||||||
_userList:
|
_userList:
|
||||||
chooseList: "Выберите список"
|
chooseList: "Выберите список"
|
||||||
|
clicker: "Счётчик щелчков"
|
||||||
_cw:
|
_cw:
|
||||||
hide: "Спрятать"
|
hide: "Спрятать"
|
||||||
show: "Показать еще"
|
show: "Показать еще"
|
||||||
@@ -1578,12 +1691,13 @@ _profile:
|
|||||||
changeAvatar: "Поменять аватар"
|
changeAvatar: "Поменять аватар"
|
||||||
changeBanner: "Поменять изображение в шапке"
|
changeBanner: "Поменять изображение в шапке"
|
||||||
_exportOrImport:
|
_exportOrImport:
|
||||||
allNotes: "Все записи\n"
|
allNotes: "Все заметки\n"
|
||||||
|
favoritedNotes: "Избранное"
|
||||||
followingList: "Подписки"
|
followingList: "Подписки"
|
||||||
muteList: "Скрытые"
|
muteList: "Скрытые"
|
||||||
blockingList: "Заблокированные"
|
blockingList: "Заблокированные"
|
||||||
userLists: "Списки"
|
userLists: "Списки"
|
||||||
excludeMutingUsers: "За исключением заглушенных пользователей"
|
excludeMutingUsers: "За исключением скрытых пользователей"
|
||||||
excludeInactiveUsers: "Без неактивных учётных записей"
|
excludeInactiveUsers: "Без неактивных учётных записей"
|
||||||
_charts:
|
_charts:
|
||||||
federation: "Федерация"
|
federation: "Федерация"
|
||||||
@@ -1687,6 +1801,8 @@ _notification:
|
|||||||
youReceivedFollowRequest: "У вас новый запрос на подписку."
|
youReceivedFollowRequest: "У вас новый запрос на подписку."
|
||||||
yourFollowRequestAccepted: "Ваш запрос на подписку одобрен."
|
yourFollowRequestAccepted: "Ваш запрос на подписку одобрен."
|
||||||
youWereInvitedToGroup: "Вы приглашены в группу."
|
youWereInvitedToGroup: "Вы приглашены в группу."
|
||||||
|
pollEnded: "Подведены окончательные итоги опроса"
|
||||||
|
emptyPushNotificationMessage: "Обновлены push-уведомления"
|
||||||
achievementEarned: "Получено достижение"
|
achievementEarned: "Получено достижение"
|
||||||
_types:
|
_types:
|
||||||
all: "Все"
|
all: "Все"
|
||||||
@@ -1696,11 +1812,13 @@ _notification:
|
|||||||
renote: "Репосты"
|
renote: "Репосты"
|
||||||
quote: "Цитаты"
|
quote: "Цитаты"
|
||||||
reaction: "Реакции"
|
reaction: "Реакции"
|
||||||
|
pollEnded: "Окончания опросов"
|
||||||
receiveFollowRequest: "Получен запрос на подписку"
|
receiveFollowRequest: "Получен запрос на подписку"
|
||||||
followRequestAccepted: "Запрос на подписку одобрен"
|
followRequestAccepted: "Запрос на подписку одобрен"
|
||||||
groupInvited: "Приглашение в группы"
|
groupInvited: "Приглашение в группы"
|
||||||
app: "Уведомления из приложений"
|
app: "Уведомления из приложений"
|
||||||
_actions:
|
_actions:
|
||||||
|
followBack: "отвечает взаимной подпиской"
|
||||||
reply: "Ответить"
|
reply: "Ответить"
|
||||||
renote: "Репост"
|
renote: "Репост"
|
||||||
_deck:
|
_deck:
|
||||||
@@ -1714,7 +1832,12 @@ _deck:
|
|||||||
swapDown: "Переставить ниже"
|
swapDown: "Переставить ниже"
|
||||||
stackLeft: "В столбик влево"
|
stackLeft: "В столбик влево"
|
||||||
popRight: "Из столбика вправо"
|
popRight: "Из столбика вправо"
|
||||||
profile: "Профиль"
|
profile: "Расстановка"
|
||||||
|
newProfile: "Новая расстановка"
|
||||||
|
deleteProfile: "Удаление расстановки"
|
||||||
|
introduction: "Создайте идеальный интерфейс расставляя колонки как угодно"
|
||||||
|
introduction2: "Чтобы добавлять колонки в любом месте, жмите «+» справа экрана."
|
||||||
|
widgetsIntroduction: "Чтобы добавлять виджеты, выбирайте «Редактировать виджеты» в меню колонки."
|
||||||
_columns:
|
_columns:
|
||||||
main: "Основная"
|
main: "Основная"
|
||||||
widgets: "Виджеты"
|
widgets: "Виджеты"
|
||||||
@@ -1722,5 +1845,6 @@ _deck:
|
|||||||
tl: "Лента"
|
tl: "Лента"
|
||||||
antenna: "Антенны"
|
antenna: "Антенны"
|
||||||
list: "Списки"
|
list: "Списки"
|
||||||
|
channel: "Каналы"
|
||||||
mentions: "Упоминания"
|
mentions: "Упоминания"
|
||||||
direct: "Личное"
|
direct: "Личное"
|
||||||
|
@@ -1545,5 +1545,6 @@ _deck:
|
|||||||
tl: "Časová os"
|
tl: "Časová os"
|
||||||
antenna: "Antény"
|
antenna: "Antény"
|
||||||
list: "Zoznam"
|
list: "Zoznam"
|
||||||
|
channel: "Kanály"
|
||||||
mentions: "Zmienky"
|
mentions: "Zmienky"
|
||||||
direct: "Priame poznámky"
|
direct: "Priame poznámky"
|
||||||
|
@@ -129,6 +129,7 @@ unblockConfirm: "คุณแน่ใจแล้วเหรอ? ว่าต
|
|||||||
suspendConfirm: "นายแน่ใจแล้วเหรอว่าต้องการระงับบัญชีนี้อ่ะ?"
|
suspendConfirm: "นายแน่ใจแล้วเหรอว่าต้องการระงับบัญชีนี้อ่ะ?"
|
||||||
unsuspendConfirm: "นายแน่ใจแล้วหรอ? ว่าต้องการยกเลิกการระงับบัญชีนี้"
|
unsuspendConfirm: "นายแน่ใจแล้วหรอ? ว่าต้องการยกเลิกการระงับบัญชีนี้"
|
||||||
selectList: "เลือกรายการ"
|
selectList: "เลือกรายการ"
|
||||||
|
selectChannel: "เลือกแชนแนล"
|
||||||
selectAntenna: "เลือกเสาอากาศ"
|
selectAntenna: "เลือกเสาอากาศ"
|
||||||
selectWidget: "เลือกวิดเจ็ต"
|
selectWidget: "เลือกวิดเจ็ต"
|
||||||
editWidgets: "แก้ไขวิดเจ็ต"
|
editWidgets: "แก้ไขวิดเจ็ต"
|
||||||
@@ -942,14 +943,239 @@ achievements: "ความสำเร็จ"
|
|||||||
_achievements:
|
_achievements:
|
||||||
earnedAt: "ได้รับเมื่อ"
|
earnedAt: "ได้รับเมื่อ"
|
||||||
_types:
|
_types:
|
||||||
|
_notes1:
|
||||||
|
title: "เพียงแค่ตั้งค่า 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: "มือใหม่ I"
|
||||||
|
description: "เข้าสู่ระบบเป็นเวลารวม 3 วัน"
|
||||||
|
flavor: "เริ่มตั้งแต่วันนี้ เรียกฉันว่ามิสคิสต์"
|
||||||
|
_login7:
|
||||||
|
title: "มือใหม่ II"
|
||||||
|
description: "เข้าสู่ระบบเป็นเวลารวม 7 วัน"
|
||||||
|
flavor: "รู้สึกเหมือนคุณได้แขวนของสิ่งต่างๆ หรือยังคะ?"
|
||||||
|
_login15:
|
||||||
|
title: "มือใหม่ III"
|
||||||
|
description: "เข้าสู่ระบบเป็นเวลารวม 15 วัน"
|
||||||
|
_login30:
|
||||||
|
title: "มิสคิสท์ I"
|
||||||
|
description: "เข้าสู่ระบบเป็นเวลารวม 30 วัน"
|
||||||
|
_login60:
|
||||||
|
title: "มิสคิสท์ II"
|
||||||
|
description: "เข้าสู่ระบบเป็นเวลารวม 60 วัน"
|
||||||
|
_login100:
|
||||||
|
title: "มิสคิสท์ III"
|
||||||
|
description: "เข้าสู่ระบบเป็นเวลารวม 100 วัน"
|
||||||
|
flavor: "ความรุนแรง Misskist"
|
||||||
|
_login200:
|
||||||
|
title: "ลูกค้าประจำ I"
|
||||||
|
description: "เข้าสู่ระบบเป็นเวลารวม 200 วัน"
|
||||||
|
_login300:
|
||||||
|
title: "ลูกค้าประจำ II"
|
||||||
|
description: "เข้าสู่ระบบเป็นเวลารวม 300 วัน"
|
||||||
|
_login400:
|
||||||
|
title: "ลูกค้าประจำ III"
|
||||||
|
description: "เข้าสู่ระบบเป็นเวลารวม 400 วัน"
|
||||||
|
_login500:
|
||||||
|
title: "ผู้เชี่ยวชาญ I"
|
||||||
|
description: "เข้าสู่ระบบเป็นเวลารวม 500 วัน"
|
||||||
|
flavor: "เพื่อนของผมนะมักจะกล่าวว่าผมนะชอบจดโน้ต"
|
||||||
|
_login600:
|
||||||
|
title: "ผู้เชี่ยวชาญ II"
|
||||||
|
description: "เข้าสู่ระบบเป็นเวลารวม 600 วัน"
|
||||||
|
_login700:
|
||||||
|
title: "ผู้เชี่ยวชาญ III"
|
||||||
|
description: "เข้าสู่ระบบเป็นเวลารวม 700 วัน"
|
||||||
|
_login800:
|
||||||
|
title: "ปรมาจารย์ด้านโน้ต I"
|
||||||
|
description: "เข้าสู่ระบบเป็นเวลารวม 800 วัน"
|
||||||
|
_login900:
|
||||||
|
title: "ปรมาจารย์ด้านโน้ต II"
|
||||||
|
description: "เข้าสู่ระบบเป็นเวลารวม 900 วัน"
|
||||||
|
_login1000:
|
||||||
|
title: "ปรมาจารย์ด้านโน้ต III"
|
||||||
|
description: "เข้าสู่ระบบเป็นเวลารวม 1,000 วัน"
|
||||||
|
flavor: "ขอบคุณที่ใช้ Misskey นะ !"
|
||||||
|
_noteClipped1:
|
||||||
|
title: "จะต้อง... คลิป..."
|
||||||
|
description: "คลิปโน้ตตัวแรกของคุณ"
|
||||||
|
_noteFavorited1:
|
||||||
|
title: "สตาร์เกเซอร์"
|
||||||
|
description: "ชื่นชอบโน้ตแรกของคุณ"
|
||||||
|
_myNoteFavorited1:
|
||||||
|
title: "แสวงหาดวงดาว"
|
||||||
|
description: "มีคนอื่นๆที่ชื่นชอบหนึ่งในโน้ตของคุณ"
|
||||||
|
_profileFilled:
|
||||||
|
title: "เตรียมไว้อย่างดี"
|
||||||
|
description: "ตั้งค่าโปรไฟล์ของคุณ"
|
||||||
|
_markedAsCat:
|
||||||
|
title: "ฉันเป็นแมว"
|
||||||
|
description: "ทำเครื่องหมายบัญชีของคุณว่าเป็นแมว"
|
||||||
|
flavor: "ฉันจะให้ชื่อคุณภายหลังนะ"
|
||||||
|
_following1:
|
||||||
|
title: "กำลังติดตามผู้ใช้คนแรกของคุณ"
|
||||||
|
description: "ติดตามผู้ใช้"
|
||||||
|
_following10:
|
||||||
|
title: "ทำต่อไป... ทำต่อไป..."
|
||||||
|
description: "ติดตาม 10 บัญชีผู้ใช้"
|
||||||
|
_following50:
|
||||||
|
title: "มีเพื่อนมากมาย"
|
||||||
|
description: "ติดตาม 50 บัญชี"
|
||||||
|
_following100:
|
||||||
|
title: "เพื่อน 100 คน"
|
||||||
|
description: "ติดตาม 100 บัญชี"
|
||||||
|
_following300:
|
||||||
|
title: "เพื่อนโอเวอร์โหลด"
|
||||||
|
description: "ติดตาม 300 บัญชี"
|
||||||
|
_followers1:
|
||||||
|
title: "ผู้ติดตามคนแรก"
|
||||||
|
description: "ได้รับ 1 ผู้ติดตาม"
|
||||||
|
_followers10:
|
||||||
|
title: "ติดตามฉัน!"
|
||||||
|
description: "ได้รับ 10 คนผู้ติดตาม"
|
||||||
|
_followers50:
|
||||||
|
title: "มากันเป็นฝูง"
|
||||||
|
description: "ได้รับ 50 ผู้ติดตาม"
|
||||||
_followers100:
|
_followers100:
|
||||||
title: "บุคคลที่เป็นที่นิยม"
|
title: "บุคคลที่เป็นที่นิยม"
|
||||||
|
description: "ได้รับ 100 ผู้ติดตาม"
|
||||||
|
_followers300:
|
||||||
|
title: "กรุณาสร้างบรรทัดเดียวนะคะ"
|
||||||
|
description: "ได้รับ 300 คนผู้ติดตาม"
|
||||||
_followers500:
|
_followers500:
|
||||||
title: "เสาสัญญาณ"
|
title: "เสาสัญญาณ"
|
||||||
|
description: "ได้รับ 500 คนผู้ติดตาม"
|
||||||
|
_followers1000:
|
||||||
|
title: "ผู้ทรงอิทธิพล"
|
||||||
|
description: "ได้รับ 1,000 ผู้ติดตาม"
|
||||||
|
_collectAchievements30:
|
||||||
|
title: "นักสะสมความสำเร็จ"
|
||||||
|
description: "ได้รับความสำเร็จ 30 ครั้ง"
|
||||||
|
_viewAchievements3min:
|
||||||
|
title: "ชอบบรรลุผลสําเร็จ"
|
||||||
|
description: "มองดูรายการความสำเร็จของคุณเป็นเวลาอย่างน้อย 3 นาที"
|
||||||
_iLoveMisskey:
|
_iLoveMisskey:
|
||||||
title: "ฉันรัก Misskey"
|
title: "ฉันรัก Misskey"
|
||||||
|
description: "โพสต์ \"I ❤ #Misskey\""
|
||||||
|
flavor: "ทีมผู้พัฒนา Misskey ได้ขอบคุณสำหรับการสนับสนุนของคุณ!"
|
||||||
|
_foundTreasure:
|
||||||
|
title: "ล่าสมบัติ"
|
||||||
|
description: "คุณพบสมบัติที่ซ่อนอยู่"
|
||||||
|
_client30min:
|
||||||
|
title: "พักผ่อนสักหน่อย"
|
||||||
|
description: "ใช้เวลา 30 นาทีบน Misskey"
|
||||||
|
_noteDeletedWithin1min:
|
||||||
|
title: "ไม่เป็นไร"
|
||||||
|
description: "ลบโน้ตภายในหนึ่งนาทีหลังจากที่โพสต์"
|
||||||
|
_postedAtLateNight:
|
||||||
|
title: "กลางคืน"
|
||||||
|
description: "โพสต์โน้ตตอนดึกๆ"
|
||||||
|
flavor: "ได้เวลาเข้านอนแล้วนะ"
|
||||||
|
_postedAt0min0sec:
|
||||||
|
title: "นาฬิกาพูดได้"
|
||||||
|
description: "โพสต์บนโน้ตเมื่อเวลา 00:00 น."
|
||||||
|
flavor: "คลิก คลิก คลิก แกล๊งๆ"
|
||||||
|
_selfQuote:
|
||||||
|
title: "อ้างอิงตนเอง"
|
||||||
|
description: "อ้างโน้ตย่อของคุณเอง"
|
||||||
|
_htl20npm:
|
||||||
|
title: "ไทม์ไลน์ไหล"
|
||||||
|
description: "มีการทำความเร็วของไทม์ไลน์ที่บ้านของคุณเกิน 20 npm (โน้ตต่อนาที)"
|
||||||
|
_viewInstanceChart:
|
||||||
|
title: "วิเคราะห์"
|
||||||
|
description: "ดูแผนภูมิอินสแตนซ์ของคุณ"
|
||||||
|
_outputHelloWorldOnScratchpad:
|
||||||
|
title: "หวัดดีชาวโลก!"
|
||||||
|
description: "เอาพุต \"hello world\" ใน Scratchpad"
|
||||||
|
_open3windows:
|
||||||
|
title: "มัลติวินโดว์"
|
||||||
|
description: "มีการเปิดหน้าต่างอย่างน้อย 3 หน้าต่างพร้อมกัน"
|
||||||
_driveFolderCircularReference:
|
_driveFolderCircularReference:
|
||||||
title: "อ้างอิงวงจร"
|
title: "อ้างอิงวงจร"
|
||||||
|
description: "พยายามสร้างโฟลเดอร์ที่ซ้อนกันแบบวนซ้ำในไดรฟ์"
|
||||||
|
_reactWithoutRead:
|
||||||
|
title: "คุณอ่านมันจริงๆหรือเปล่า?"
|
||||||
|
description: "มีการโต้ตอบกับโน้ตที่มีความยาวมากกว่า 100 ตัวอักษรภายใน 3 วินาทีหลังจากที่โพสต์"
|
||||||
|
_clickedClickHere:
|
||||||
|
title: "คลิ๊กที่นี่"
|
||||||
|
description: "คุณได้คลิกที่นี่"
|
||||||
|
_justPlainLucky:
|
||||||
|
title: "แค่ลัคกี้ธรรมดา"
|
||||||
|
description: "มีโอกาสที่จะได้รับด้วยความน่าจะเป็นไปได้ 0.005% ทุก ๆ 10 วินาที"
|
||||||
|
_setNameToSyuilo:
|
||||||
|
title: "พระเจ้าคอมเพล็กซ์"
|
||||||
|
description: "ตั้งชื่อของคุณเป็น \"syuilo\""
|
||||||
|
_passedSinceAccountCreated1:
|
||||||
|
title: "ครบรอบหนึ่งปี"
|
||||||
|
description: "ผ่านไปหนึ่งปีแล้วนะตั้งแต่บัญชีของคุณถูกสร้างขึ้นมาน่ะ"
|
||||||
|
_passedSinceAccountCreated2:
|
||||||
|
title: "ครบรอบสองปี"
|
||||||
|
description: "ผ่านไปสองปีแล้วนะตั้งแต่บัญชีของคุณถูกสร้างขึ้นมาน่ะ"
|
||||||
|
_passedSinceAccountCreated3:
|
||||||
|
title: "ครบรอบสามปี"
|
||||||
|
description: "ผ่านไปสามปีแล้วนะตั้งแต่บัญชีของคุณถูกสร้างขึ้นมาน่ะ"
|
||||||
|
_loggedInOnBirthday:
|
||||||
|
title: "สุขสันต์วันเกิด"
|
||||||
|
description: "เข้าสู่ระบบในวันเกิดของคุณ"
|
||||||
|
_loggedInOnNewYearsDay:
|
||||||
|
title: "สวัสดีปีใหม่!"
|
||||||
|
description: "เข้าสู่ระบบในวันแรกของปี"
|
||||||
|
flavor: "อีกปีที่ยอดเยี่ยมในโอกาสนี้เลย"
|
||||||
|
_cookieClicked:
|
||||||
|
title: "เกมที่คุณคลิกที่คุกกี้"
|
||||||
|
description: "คลิกคุกกี้"
|
||||||
|
flavor: "เดี๋ยวก่อนนะ คุณอยู่ในเว็บไซต์ที่ถูกต้องแน่อย่างงั้นเหรอ?"
|
||||||
|
_brainDiver:
|
||||||
|
title: "Brain Diver"
|
||||||
|
description: "โพสต์ลิงก์ไปยัง Brain Diver"
|
||||||
|
flavor: "Misskey-Misskey La-Tu-Ma"
|
||||||
_role:
|
_role:
|
||||||
new: "บทบาทใหม่"
|
new: "บทบาทใหม่"
|
||||||
edit: "แก้ไขบทบาท"
|
edit: "แก้ไขบทบาท"
|
||||||
@@ -957,7 +1183,7 @@ _role:
|
|||||||
description: "คำอธิบายบทบาท"
|
description: "คำอธิบายบทบาท"
|
||||||
permission: "สิทธิ์ตามบทบาท"
|
permission: "สิทธิ์ตามบทบาท"
|
||||||
descriptionOfPermission: "<b>ผู้ดูแลกลั่นกรองเนื้อหา</b> สามารถดำเนินการดูแลขั้นพื้นฐานได้นะ\n<b>ผู้ดูแลระบบ</b> สามารถเปลี่ยนการตั้งค่าทั้งหมดของอินสแตนซ์ได้นะ"
|
descriptionOfPermission: "<b>ผู้ดูแลกลั่นกรองเนื้อหา</b> สามารถดำเนินการดูแลขั้นพื้นฐานได้นะ\n<b>ผู้ดูแลระบบ</b> สามารถเปลี่ยนการตั้งค่าทั้งหมดของอินสแตนซ์ได้นะ"
|
||||||
assignTarget: "กำหนดเป้าหมาย"
|
assignTarget: "มอบหมาย"
|
||||||
descriptionOfAssignTarget: "<b>แมนนวล</b> เพื่อเปลี่ยนผู้ที่เป็นส่วนหนึ่งของบทบาทนี้และใครที่ไม่ใช่ด้วยตนเอง\n<b>เงื่อนไข</b> เพื่อให้ผู้ใช้ได้รับการกำหนดและนำออกจากบทบาทนี้โดยอัตโนมัติตามเงื่อนไขชุดหนึ่ง"
|
descriptionOfAssignTarget: "<b>แมนนวล</b> เพื่อเปลี่ยนผู้ที่เป็นส่วนหนึ่งของบทบาทนี้และใครที่ไม่ใช่ด้วยตนเอง\n<b>เงื่อนไข</b> เพื่อให้ผู้ใช้ได้รับการกำหนดและนำออกจากบทบาทนี้โดยอัตโนมัติตามเงื่อนไขชุดหนึ่ง"
|
||||||
manual: "ปรับเอง"
|
manual: "ปรับเอง"
|
||||||
conditional: "มีเงื่อนไข"
|
conditional: "มีเงื่อนไข"
|
||||||
@@ -970,6 +1196,9 @@ _role:
|
|||||||
baseRole: "บทบาทพื้นฐาน"
|
baseRole: "บทบาทพื้นฐาน"
|
||||||
useBaseValue: "ใช้บทบาทพื้นฐานเริ่มต้น"
|
useBaseValue: "ใช้บทบาทพื้นฐานเริ่มต้น"
|
||||||
chooseRoleToAssign: "เลือกบทบาทที่ต้องการกำหนด"
|
chooseRoleToAssign: "เลือกบทบาทที่ต้องการกำหนด"
|
||||||
|
iconUrl: "ไอคอน URL"
|
||||||
|
asBadge: "แสดงเป็นตรา"
|
||||||
|
descriptionOfAsBadge: "ไอคอนของบทบาทนี้จะปรากฏถัดจากชื่อผู้ใช้ของผู้ใช้งานด้วยบทบาทนี้ถ้าหากเปิดใช้งาน"
|
||||||
canEditMembersByModerator: "อนุญาตให้ผู้ดูแลแก้ไขสมาชิก"
|
canEditMembersByModerator: "อนุญาตให้ผู้ดูแลแก้ไขสมาชิก"
|
||||||
descriptionOfCanEditMembersByModerator: "เมื่อเปิดใช้ ผู้ดูแลนอกเหนือจากผู้ดูแลระบบแล้ว จะสามารถกำหนดและยกเลิกการมอบหมายบทบาทนี้ให้กับผู้ใช้ได้ เมื่อปิด เฉพาะผู้ดูแลระบบเท่านั้นที่จะสามารถกำหนดผู้ใช้ได้นะ"
|
descriptionOfCanEditMembersByModerator: "เมื่อเปิดใช้ ผู้ดูแลนอกเหนือจากผู้ดูแลระบบแล้ว จะสามารถกำหนดและยกเลิกการมอบหมายบทบาทนี้ให้กับผู้ใช้ได้ เมื่อปิด เฉพาะผู้ดูแลระบบเท่านั้นที่จะสามารถกำหนดผู้ใช้ได้นะ"
|
||||||
priority: "ลำดับความสำคัญ"
|
priority: "ลำดับความสำคัญ"
|
||||||
@@ -1641,5 +1870,6 @@ _deck:
|
|||||||
tl: "ไทม์ไลน์"
|
tl: "ไทม์ไลน์"
|
||||||
antenna: "เสาอากาศ"
|
antenna: "เสาอากาศ"
|
||||||
list: "รายการ"
|
list: "รายการ"
|
||||||
|
channel: "แชนแนล"
|
||||||
mentions: "พูดถึง"
|
mentions: "พูดถึง"
|
||||||
direct: "ไดเร็ค"
|
direct: "ไดเร็ค"
|
||||||
|
@@ -529,7 +529,7 @@ state: "Стан"
|
|||||||
sort: "Сортування"
|
sort: "Сортування"
|
||||||
ascendingOrder: "За зростанням"
|
ascendingOrder: "За зростанням"
|
||||||
descendingOrder: "За спаданням"
|
descendingOrder: "За спаданням"
|
||||||
scratchpad: "Чернетка"
|
scratchpad: "Scratchpad"
|
||||||
scratchpadDescription: "Scratchpad надає середовище для експериментів з AiScript. Ви можете писати, виконувати його і тестувати взаємодію з Misskey."
|
scratchpadDescription: "Scratchpad надає середовище для експериментів з AiScript. Ви можете писати, виконувати його і тестувати взаємодію з Misskey."
|
||||||
output: "Вихід"
|
output: "Вихід"
|
||||||
script: "Скрипт"
|
script: "Скрипт"
|
||||||
@@ -688,7 +688,7 @@ pageLikesCount: "Кількість отриманих вподобань сто
|
|||||||
pageLikedCount: "Кількість вподобаних сторінок"
|
pageLikedCount: "Кількість вподобаних сторінок"
|
||||||
contact: "Контакт"
|
contact: "Контакт"
|
||||||
useSystemFont: "Використовувати стандартний шрифт системи"
|
useSystemFont: "Використовувати стандартний шрифт системи"
|
||||||
clips: "Добірка"
|
clips: "Добірки"
|
||||||
experimentalFeatures: "Експериментальні функції"
|
experimentalFeatures: "Експериментальні функції"
|
||||||
developer: "Розробник"
|
developer: "Розробник"
|
||||||
makeExplorable: "Зробіть обліковий запис видимим у розділі \"Огляд\""
|
makeExplorable: "Зробіть обліковий запис видимим у розділі \"Огляд\""
|
||||||
@@ -899,6 +899,222 @@ unlike: "Не вподобати"
|
|||||||
numberOfLikes: "Вподобання"
|
numberOfLikes: "Вподобання"
|
||||||
show: "Відображення"
|
show: "Відображення"
|
||||||
color: "Колір"
|
color: "Колір"
|
||||||
|
achievements: "Досягнення"
|
||||||
|
_achievements:
|
||||||
|
earnedAt: "Відкрито"
|
||||||
|
_types:
|
||||||
|
_notes1:
|
||||||
|
title: "Привіт, Misskey!"
|
||||||
|
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: "Новачок I"
|
||||||
|
description: "3 дні користування загально"
|
||||||
|
flavor: "Відсьогодні називайте мене \"Місскіст\""
|
||||||
|
_login7:
|
||||||
|
title: "Новачок II"
|
||||||
|
description: "7 днів користування загально"
|
||||||
|
flavor: "Ви звикли до цього?"
|
||||||
|
_login15:
|
||||||
|
title: "Новачок III"
|
||||||
|
description: "15 днів користування загально"
|
||||||
|
_login30:
|
||||||
|
title: "Міскієць I"
|
||||||
|
description: "30 днів користування загально"
|
||||||
|
_login60:
|
||||||
|
title: "Міскієць II"
|
||||||
|
description: "60 днів користування загально"
|
||||||
|
_login100:
|
||||||
|
title: "Міскієць III"
|
||||||
|
description: "100 днів користування загально"
|
||||||
|
flavor: "Цей юзер лютий місскіст"
|
||||||
|
_login200:
|
||||||
|
title: "Завсідник I"
|
||||||
|
description: "200 днів користування загально"
|
||||||
|
_login300:
|
||||||
|
title: "Завсідник II"
|
||||||
|
description: "300 днів користування загально"
|
||||||
|
_login400:
|
||||||
|
title: "Завсідник III"
|
||||||
|
description: "400 днів користування загально"
|
||||||
|
_login500:
|
||||||
|
title: "Ветеран I"
|
||||||
|
description: "500 днів користування загально"
|
||||||
|
flavor: "Meine Kameraden, ich liebe sie, die Notizen."
|
||||||
|
_login600:
|
||||||
|
title: "Ветеран II"
|
||||||
|
description: "600 днів користування загально"
|
||||||
|
_login700:
|
||||||
|
title: "Ветеран III"
|
||||||
|
description: "700 днів користування загально"
|
||||||
|
_login800:
|
||||||
|
title: "Майстер нотаток I"
|
||||||
|
description: "800 днів користування загально"
|
||||||
|
_login900:
|
||||||
|
title: "Майстер нотаток II"
|
||||||
|
description: "900 днів користування загально"
|
||||||
|
_login1000:
|
||||||
|
title: "Майстер нотаток III"
|
||||||
|
description: "1000 днів користування загально"
|
||||||
|
flavor: "Дякуємо, що користуєтеся Misskey!"
|
||||||
|
_noteClipped1:
|
||||||
|
title: "Не можна не зберегти"
|
||||||
|
description: "Перша нотатка у добірці"
|
||||||
|
_noteFavorited1:
|
||||||
|
title: "Дивитися на зірки"
|
||||||
|
_myNoteFavorited1:
|
||||||
|
title: "У пошуках зірок"
|
||||||
|
_profileFilled:
|
||||||
|
title: "Повна готовність"
|
||||||
|
description: "Профіль заповнено"
|
||||||
|
_markedAsCat:
|
||||||
|
title: "Я кіт"
|
||||||
|
description: "Позначено як акаунт кота"
|
||||||
|
flavor: "Я дам тобі ім'я пізніше"
|
||||||
|
_following1:
|
||||||
|
title: "Перша підписка"
|
||||||
|
_following10:
|
||||||
|
title: "Продовжуй, продовжуй"
|
||||||
|
_following50:
|
||||||
|
title: "Багато друзів"
|
||||||
|
description: "Кількість підписок сягнула 50"
|
||||||
|
_following100:
|
||||||
|
title: "100 друзів"
|
||||||
|
description: "Кількість підписок сягнула 100"
|
||||||
|
_following300:
|
||||||
|
title: "Надлишок друзів"
|
||||||
|
description: "Кількість підписок сягнула 300"
|
||||||
|
_followers1:
|
||||||
|
title: "Перший підписник"
|
||||||
|
description: "З'явився перший підписник"
|
||||||
|
_followers10:
|
||||||
|
title: "Follow me!"
|
||||||
|
description: "Кількість підписників досягла 10"
|
||||||
|
_followers50:
|
||||||
|
description: "Кількість підписників досягла 50"
|
||||||
|
_followers100:
|
||||||
|
title: "Популярна особа"
|
||||||
|
description: "Кількість підписників досягла 100"
|
||||||
|
_followers300:
|
||||||
|
title: "Ставайте в чергу"
|
||||||
|
description: "Кількість підписників досягла 300"
|
||||||
|
_followers500:
|
||||||
|
title: "Радіовежа"
|
||||||
|
description: "Кількість підписників досягла 500"
|
||||||
|
_followers1000:
|
||||||
|
title: "Інфлюенсер"
|
||||||
|
description: "Кількість підписників досягла 1000"
|
||||||
|
_collectAchievements30:
|
||||||
|
title: "Збирач досягнень"
|
||||||
|
description: "Отримано 30 досягнень"
|
||||||
|
_viewAchievements3min:
|
||||||
|
title: "Шанувальник досягнень"
|
||||||
|
description: "Переглядати список досягнень принаймні 3 хвилини"
|
||||||
|
_iLoveMisskey:
|
||||||
|
title: "I Love Misskey"
|
||||||
|
description: "Відправлено \"I ❤ #Misskey\""
|
||||||
|
flavor: "Дякуємо вам, що користуєтесь Misskey! – команда розробників"
|
||||||
|
_foundTreasure:
|
||||||
|
title: "Пошуки скарбів"
|
||||||
|
description: "Ви знайшли прихований скарб"
|
||||||
|
_client30min:
|
||||||
|
title: "Коротка перерва"
|
||||||
|
description: "З моменту запуску клієнта минуло 30 хвилин"
|
||||||
|
_noteDeletedWithin1min:
|
||||||
|
title: "Не зважай"
|
||||||
|
description: "Допис видалено протягом 1 хвилини після публікації"
|
||||||
|
_postedAtLateNight:
|
||||||
|
title: "Нічне життя"
|
||||||
|
description: "Відправити нотатку посеред ночі"
|
||||||
|
flavor: "Час лягати спати"
|
||||||
|
_postedAt0min0sec:
|
||||||
|
title: "Сигнал часу"
|
||||||
|
description: "Відправити нотатку о 00:00"
|
||||||
|
_selfQuote:
|
||||||
|
title: "Самопосилання"
|
||||||
|
description: "Процитувати власну нотатку"
|
||||||
|
_htl20npm:
|
||||||
|
title: "Плинна стрічка"
|
||||||
|
description: "Перевищити швидкість домашньої стрічки 20npm (нотаток на хвилину)"
|
||||||
|
_viewInstanceChart:
|
||||||
|
title: "Аналітик"
|
||||||
|
_outputHelloWorldOnScratchpad:
|
||||||
|
title: "Hello, world!"
|
||||||
|
description: "Вивести \"hello world\" у Скретчпаді"
|
||||||
|
_clickedClickHere:
|
||||||
|
title: "Натисніть тут"
|
||||||
|
description: "Натиснуто тут"
|
||||||
|
_justPlainLucky:
|
||||||
|
title: "Просто вдача"
|
||||||
|
description: "Можна отримати з ймовірністю 0,01% кожні 10 секунд"
|
||||||
|
_setNameToSyuilo:
|
||||||
|
title: "Комплекс бога"
|
||||||
|
description: "Встановлено ім'я \"syuilo\""
|
||||||
|
_passedSinceAccountCreated1:
|
||||||
|
title: "Перша річниця"
|
||||||
|
description: "Минув рік з моменту створення акаунта"
|
||||||
|
_passedSinceAccountCreated2:
|
||||||
|
title: "Друга річниця"
|
||||||
|
description: "Минуло 2 роки з моменту створення акаунта"
|
||||||
|
_passedSinceAccountCreated3:
|
||||||
|
title: "Третя річниця"
|
||||||
|
description: "Минуло 3 роки з моменту створення акаунта"
|
||||||
|
_loggedInOnBirthday:
|
||||||
|
title: "З Днем народження!"
|
||||||
|
description: "Увійти у свій день народження"
|
||||||
|
_loggedInOnNewYearsDay:
|
||||||
|
title: "З Новим роком!"
|
||||||
|
description: "Увійшли в перший день року"
|
||||||
|
_brainDiver:
|
||||||
|
title: "Brain Diver"
|
||||||
|
description: "Відправити посилання на \"Brain Diver\""
|
||||||
|
flavor: "Misskey-Misskey La-Tu-Ma"
|
||||||
_role:
|
_role:
|
||||||
priority: "Пріоритет"
|
priority: "Пріоритет"
|
||||||
_priority:
|
_priority:
|
||||||
@@ -1166,12 +1382,12 @@ _tutorial:
|
|||||||
step1_1: "Ласкаво просимо!"
|
step1_1: "Ласкаво просимо!"
|
||||||
step1_2: "Ця сторінка має назву \"стрічка подій\". На ній з'являються записи користувачів на яких ви підписані."
|
step1_2: "Ця сторінка має назву \"стрічка подій\". На ній з'являються записи користувачів на яких ви підписані."
|
||||||
step1_3: "Наразі ваша стрічка порожня, оскільки ви ще не написали жодної нотатки і не підписані на інших."
|
step1_3: "Наразі ваша стрічка порожня, оскільки ви ще не написали жодної нотатки і не підписані на інших."
|
||||||
step2_1: "Перш ніж зробити запис або підписатись на когось, спочатку заповніть свій обліковий запис."
|
step2_1: "Перш ніж зробити запис або підписатись на когось, заповніть свій профіль."
|
||||||
step2_2: "Надання деякої інформації про себе дозволить іншим користувачам підписатись на вас."
|
step2_2: "Надання деякої інформації про себе допоможе іншим користувачам вирішити підписатись на вас."
|
||||||
step3_1: "Ви успішно налаштували свій обліковий запис?"
|
step3_1: "Ви успішно налаштували свій обліковий запис?"
|
||||||
step3_2: "Наступним кроком є написання нотатки. Це можна зробити, натиснувши зображення олівця на екрані."
|
step3_2: "Наступним кроком є написання нотатки. Це можна зробити, натиснувши зображення олівця на екрані."
|
||||||
step3_3: "Після написання вмісту ви можете опублікувати його, натиснувши кнопку у верхньому правому куті форми."
|
step3_3: "Після написання вмісту ви можете опублікувати його, натиснувши кнопку у верхньому правому куті форми."
|
||||||
step3_4: "Не знаєте що написати? Спробуйте \"налаштовую свій msky\"!"
|
step3_4: "Не знаєте що написати? Спробуйте \"Привіт, Misskey!\""
|
||||||
step4_1: "Ви розмістили свій перший запис?"
|
step4_1: "Ви розмістили свій перший запис?"
|
||||||
step4_2: "Ура! Ваш перший запис відображається на вашій стрічці подій."
|
step4_2: "Ура! Ваш перший запис відображається на вашій стрічці подій."
|
||||||
step5_1: "Настав час оживити вашу стрічку подій підписавшись на інших користувачів."
|
step5_1: "Настав час оживити вашу стрічку подій підписавшись на інших користувачів."
|
||||||
@@ -1435,6 +1651,7 @@ _notification:
|
|||||||
youReceivedFollowRequest: "Ви отримали запит на підписку"
|
youReceivedFollowRequest: "Ви отримали запит на підписку"
|
||||||
yourFollowRequestAccepted: "Запит на підписку прийнято"
|
yourFollowRequestAccepted: "Запит на підписку прийнято"
|
||||||
youWereInvitedToGroup: "Запрошення до групи"
|
youWereInvitedToGroup: "Запрошення до групи"
|
||||||
|
achievementEarned: "Досягнення відкрито"
|
||||||
_types:
|
_types:
|
||||||
all: "Все"
|
all: "Все"
|
||||||
follow: "Підписки"
|
follow: "Підписки"
|
||||||
@@ -1472,5 +1689,6 @@ _deck:
|
|||||||
tl: "Стрічка"
|
tl: "Стрічка"
|
||||||
antenna: "Антени"
|
antenna: "Антени"
|
||||||
list: "Списки"
|
list: "Списки"
|
||||||
|
channel: "Канали"
|
||||||
mentions: "Згадки"
|
mentions: "Згадки"
|
||||||
direct: "Особисте"
|
direct: "Особисте"
|
||||||
|
@@ -1520,5 +1520,6 @@ _deck:
|
|||||||
tl: "Bảng tin"
|
tl: "Bảng tin"
|
||||||
antenna: "Trạm phát sóng"
|
antenna: "Trạm phát sóng"
|
||||||
list: "Danh sách"
|
list: "Danh sách"
|
||||||
|
channel: "Kênh"
|
||||||
mentions: "Lượt nhắc"
|
mentions: "Lượt nhắc"
|
||||||
direct: "Nhắn riêng"
|
direct: "Nhắn riêng"
|
||||||
|
@@ -956,7 +956,7 @@ _achievements:
|
|||||||
title: "满是帖子"
|
title: "满是帖子"
|
||||||
description: "发布了500篇帖子"
|
description: "发布了500篇帖子"
|
||||||
_notes1000:
|
_notes1000:
|
||||||
title: "帖子成山"
|
title: "积帖成山"
|
||||||
description: "发布了1,000篇帖子"
|
description: "发布了1,000篇帖子"
|
||||||
_notes5000:
|
_notes5000:
|
||||||
title: "帖如泉涌"
|
title: "帖如泉涌"
|
||||||
@@ -992,28 +992,189 @@ _achievements:
|
|||||||
title: "ALL YOUR NOTE ARE BELONG TO US"
|
title: "ALL YOUR NOTE ARE BELONG TO US"
|
||||||
description: "发布了100,000篇帖子"
|
description: "发布了100,000篇帖子"
|
||||||
flavor: "真的有那么多可以写的东西吗?"
|
flavor: "真的有那么多可以写的东西吗?"
|
||||||
|
_login3:
|
||||||
|
title: "初学者 I"
|
||||||
|
description: "连续登录3天"
|
||||||
|
flavor: "今天开始我就是Misskist!"
|
||||||
|
_login7:
|
||||||
|
title: "初学者 II"
|
||||||
|
description: "连续登录7天"
|
||||||
|
flavor: "您开始习惯了吗?"
|
||||||
|
_login15:
|
||||||
|
title: "初学者 III"
|
||||||
|
description: "连续登录15天"
|
||||||
|
_login30:
|
||||||
|
title: "Misskist Ⅰ"
|
||||||
|
description: "连续登录30天"
|
||||||
|
_login60:
|
||||||
|
title: "Misskist Ⅱ"
|
||||||
|
description: "连续登录60天"
|
||||||
|
_login100:
|
||||||
|
title: "Misskist Ⅲ"
|
||||||
|
description: "总登入100天"
|
||||||
|
flavor: "那个用户,是Misskist喔"
|
||||||
|
_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:
|
_login1000:
|
||||||
|
title: "帖子大师Ⅲ"
|
||||||
|
description: "总登录天数1000天"
|
||||||
flavor: "感谢您使用Misskey!"
|
flavor: "感谢您使用Misskey!"
|
||||||
|
_noteClipped1:
|
||||||
|
title: "忍不住要收藏到便签"
|
||||||
|
description: "第一次将贴文贴进便签"
|
||||||
_noteFavorited1:
|
_noteFavorited1:
|
||||||
title: "观星者"
|
title: "观星者"
|
||||||
|
description: "第一次将帖子加入收藏"
|
||||||
|
_myNoteFavorited1:
|
||||||
|
title: "想要星星"
|
||||||
|
description: "自己的帖子被其他人加入收藏了"
|
||||||
|
_profileFilled:
|
||||||
|
title: "整装待发"
|
||||||
|
description: "设置了个人资料"
|
||||||
_markedAsCat:
|
_markedAsCat:
|
||||||
title: "我是猫"
|
title: "我是猫"
|
||||||
|
description: "将账户设定为一只猫"
|
||||||
|
flavor: "还没有名字"
|
||||||
|
_following1:
|
||||||
|
title: "首次关注"
|
||||||
|
description: "第一次关注别人"
|
||||||
|
_following10:
|
||||||
|
title: "关注,跟随"
|
||||||
|
description: "关注超过10人"
|
||||||
_following50:
|
_following50:
|
||||||
title: "我的朋友很多"
|
title: "我的朋友很多"
|
||||||
|
description: "关注超过50人"
|
||||||
|
_following100:
|
||||||
|
title: "我的朋友很多"
|
||||||
|
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: "拥有超过1000名关注者"
|
||||||
|
_collectAchievements30:
|
||||||
|
title: "成就收藏家"
|
||||||
|
description: "获得超过30个成就"
|
||||||
_viewAchievements3min:
|
_viewAchievements3min:
|
||||||
|
title: "成就爱好者"
|
||||||
description: "盯着成就看三分钟"
|
description: "盯着成就看三分钟"
|
||||||
_iLoveMisskey:
|
_iLoveMisskey:
|
||||||
title: "I Love Misskey"
|
title: "I Love Misskey"
|
||||||
description: "发布\"I ❤ #Misskey\"帖子"
|
description: "发布\"I ❤ #Misskey\"帖子"
|
||||||
flavor: "感谢您使用 Misskey ! by 开发团队"
|
flavor: "感谢您使用 Misskey ! by 开发团队"
|
||||||
|
_foundTreasure:
|
||||||
|
title: "寻宝"
|
||||||
|
description: "发现了隐藏的宝藏"
|
||||||
|
_client30min:
|
||||||
|
title: "休息一下!"
|
||||||
|
description: "启动客户端超过30分钟"
|
||||||
|
_noteDeletedWithin1min:
|
||||||
|
title: "无话可说"
|
||||||
|
description: "发帖后一分钟内就将其删除"
|
||||||
|
_postedAtLateNight:
|
||||||
|
title: "夜猫子"
|
||||||
|
description: "深夜发布帖子"
|
||||||
|
flavor: "差不多该去睡了喔。"
|
||||||
|
_postedAt0min0sec:
|
||||||
|
title: "报时"
|
||||||
|
description: "在0点发布一篇帖子"
|
||||||
|
flavor: "嘣 嘣 嘣 Biu——!"
|
||||||
|
_selfQuote:
|
||||||
|
title: "自我提及"
|
||||||
|
description: "引用了自己的帖子"
|
||||||
|
_htl20npm:
|
||||||
|
title: "流动的时间线"
|
||||||
|
description: "在首页时间线的流速超过20npm"
|
||||||
|
_viewInstanceChart:
|
||||||
|
title: "分析师"
|
||||||
|
description: "查看了实例信息中的图表"
|
||||||
_outputHelloWorldOnScratchpad:
|
_outputHelloWorldOnScratchpad:
|
||||||
title: "Hello, world!"
|
title: "Hello, world!"
|
||||||
|
description: "在AiScript控制台中输出 hello world"
|
||||||
|
_open3windows:
|
||||||
|
title: "多窗口"
|
||||||
|
description: "打开了三个或更多的窗口"
|
||||||
|
_driveFolderCircularReference:
|
||||||
|
title: "循环引用"
|
||||||
|
description: "试图对网盘中的文件夹进行循环嵌套"
|
||||||
|
_reactWithoutRead:
|
||||||
|
title: "有好好读过吗?"
|
||||||
|
description: "在含有100字以上的帖子被发出三秒内做出回应"
|
||||||
|
_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:
|
_loggedInOnBirthday:
|
||||||
title: "生日快乐"
|
title: "生日快乐"
|
||||||
description: "在生日当天登录"
|
description: "在生日当天登录"
|
||||||
_loggedInOnNewYearsDay:
|
_loggedInOnNewYearsDay:
|
||||||
title: "恭贺新禧"
|
title: "恭贺新禧"
|
||||||
description: "在元旦登入"
|
description: "在元旦登入"
|
||||||
|
flavor: "今年也请对本实例多多指教!"
|
||||||
|
_cookieClicked:
|
||||||
|
title: "点击饼干小游戏"
|
||||||
|
description: "点击了可疑的饼干"
|
||||||
|
flavor: "是不是软件有问题?"
|
||||||
|
_brainDiver:
|
||||||
|
title: "Brain Diver"
|
||||||
|
description: "发布了包含Brain Diver链接的帖子"
|
||||||
|
flavor: "Misskey-Misskey La-Tu-Ma"
|
||||||
_role:
|
_role:
|
||||||
new: "创建角色"
|
new: "创建角色"
|
||||||
edit: "编辑角色"
|
edit: "编辑角色"
|
||||||
@@ -1034,6 +1195,9 @@ _role:
|
|||||||
baseRole: "基本角色"
|
baseRole: "基本角色"
|
||||||
useBaseValue: "使用基本角色的值"
|
useBaseValue: "使用基本角色的值"
|
||||||
chooseRoleToAssign: "选择要分配的角色"
|
chooseRoleToAssign: "选择要分配的角色"
|
||||||
|
iconUrl: "图标URL"
|
||||||
|
asBadge: "作为徽章显示"
|
||||||
|
descriptionOfAsBadge: "开启后,用户名旁边将会出现角色图标。"
|
||||||
canEditMembersByModerator: "允许监察者编辑成员"
|
canEditMembersByModerator: "允许监察者编辑成员"
|
||||||
descriptionOfCanEditMembersByModerator: "如果选中,监察者和管理员都能够为用户分配/取消分配角色。如果未选中,则只有管理员可以执行此操作。"
|
descriptionOfCanEditMembersByModerator: "如果选中,监察者和管理员都能够为用户分配/取消分配角色。如果未选中,则只有管理员可以执行此操作。"
|
||||||
priority: "优先级"
|
priority: "优先级"
|
||||||
@@ -1532,7 +1696,7 @@ _profile:
|
|||||||
name: "昵称"
|
name: "昵称"
|
||||||
username: "用户名"
|
username: "用户名"
|
||||||
description: "个人简介"
|
description: "个人简介"
|
||||||
youCanIncludeHashtags: "您可以包含一个哈希标签。"
|
youCanIncludeHashtags: "你可以在个人简介中包含一个#标签。"
|
||||||
metadata: "附加信息"
|
metadata: "附加信息"
|
||||||
metadataEdit: "附加信息编辑"
|
metadataEdit: "附加信息编辑"
|
||||||
metadataDescription: "最多可以在个人资料中以表格形式显示四条其他信息。"
|
metadataDescription: "最多可以在个人资料中以表格形式显示四条其他信息。"
|
||||||
@@ -1705,5 +1869,6 @@ _deck:
|
|||||||
tl: "时间线"
|
tl: "时间线"
|
||||||
antenna: "天线"
|
antenna: "天线"
|
||||||
list: "列表"
|
list: "列表"
|
||||||
|
channel: "频道"
|
||||||
mentions: "提及"
|
mentions: "提及"
|
||||||
direct: "指定用户"
|
direct: "指定用户"
|
||||||
|
@@ -240,7 +240,7 @@ removeAreYouSure: "確定要刪掉「{x}」嗎?"
|
|||||||
deleteAreYouSure: "確定要刪掉「{x}」嗎?"
|
deleteAreYouSure: "確定要刪掉「{x}」嗎?"
|
||||||
resetAreYouSure: "確定要重設嗎?"
|
resetAreYouSure: "確定要重設嗎?"
|
||||||
saved: "已儲存"
|
saved: "已儲存"
|
||||||
messaging: "傳送訊息"
|
messaging: "聊天"
|
||||||
upload: "上傳"
|
upload: "上傳"
|
||||||
keepOriginalUploading: "保留原圖"
|
keepOriginalUploading: "保留原圖"
|
||||||
keepOriginalUploadingDescription: "上傳圖片時保留原始圖片。關閉時,瀏覽器會在上傳時生成一張用於web發布的圖片。"
|
keepOriginalUploadingDescription: "上傳圖片時保留原始圖片。關閉時,瀏覽器會在上傳時生成一張用於web發布的圖片。"
|
||||||
@@ -326,15 +326,15 @@ connectService: "己連結"
|
|||||||
disconnectService: "己斷開 "
|
disconnectService: "己斷開 "
|
||||||
enableLocalTimeline: "開啟本地時間軸"
|
enableLocalTimeline: "開啟本地時間軸"
|
||||||
enableGlobalTimeline: "啟用全域時間軸"
|
enableGlobalTimeline: "啟用全域時間軸"
|
||||||
disablingTimelinesInfo: "為了方便,即使您關閉了時間線功能,管理員和審核員仍可以繼續使用。"
|
disablingTimelinesInfo: "為了方便,即使您關閉了時間線功能,管理員和審查員仍可以繼續使用。"
|
||||||
registration: "註冊"
|
registration: "註冊"
|
||||||
enableRegistration: "開啟新使用者註冊"
|
enableRegistration: "開啟新使用者註冊"
|
||||||
invite: "邀請"
|
invite: "邀請"
|
||||||
driveCapacityPerLocalAccount: "每個本地用戶的雲端空間大小"
|
driveCapacityPerLocalAccount: "每個本地用戶的雲端空間大小"
|
||||||
driveCapacityPerRemoteAccount: "每個非本地用戶的雲端容量"
|
driveCapacityPerRemoteAccount: "每個非本地用戶的雲端空間大小"
|
||||||
inMb: "以Mbps為單位"
|
inMb: "以Mbps為單位"
|
||||||
iconUrl: "圖像URL"
|
iconUrl: "圖標URL"
|
||||||
bannerUrl: "橫幅圖像URL"
|
bannerUrl: "橫幅圖片URL"
|
||||||
backgroundImageUrl: "背景圖片的來源網址 "
|
backgroundImageUrl: "背景圖片的來源網址 "
|
||||||
basicInfo: "基本資訊"
|
basicInfo: "基本資訊"
|
||||||
pinnedUsers: "置頂用戶"
|
pinnedUsers: "置頂用戶"
|
||||||
@@ -373,8 +373,8 @@ connectedTo: "您的帳戶已連接到以下社交帳戶"
|
|||||||
notesAndReplies: "貼文與回覆"
|
notesAndReplies: "貼文與回覆"
|
||||||
withFiles: "附件"
|
withFiles: "附件"
|
||||||
silence: "禁言"
|
silence: "禁言"
|
||||||
silenceConfirm: "確定要禁言此用戶嗎?"
|
silenceConfirm: "確定要靜音此使用者嗎?"
|
||||||
unsilence: "解除禁言"
|
unsilence: "解除靜音"
|
||||||
unsilenceConfirm: "確定要解除禁言嗎?"
|
unsilenceConfirm: "確定要解除禁言嗎?"
|
||||||
popularUsers: "熱門使用者"
|
popularUsers: "熱門使用者"
|
||||||
recentlyUpdatedUsers: "最近發文的使用者"
|
recentlyUpdatedUsers: "最近發文的使用者"
|
||||||
@@ -383,14 +383,14 @@ recentlyDiscoveredUsers: "最近發現的使用者"
|
|||||||
exploreUsersCount: "有{count}個使用者"
|
exploreUsersCount: "有{count}個使用者"
|
||||||
exploreFediverse: "探索聯邦世界"
|
exploreFediverse: "探索聯邦世界"
|
||||||
popularTags: "熱門標籤"
|
popularTags: "熱門標籤"
|
||||||
userList: "清單"
|
userList: "使用者清單"
|
||||||
about: "資訊"
|
about: "關於"
|
||||||
aboutMisskey: "關於 Misskey"
|
aboutMisskey: "關於 Misskey"
|
||||||
administrator: "管理員"
|
administrator: "管理員"
|
||||||
token: "權杖"
|
token: "權杖"
|
||||||
twoStepAuthentication: "兩階段驗證"
|
twoStepAuthentication: "兩階段驗證"
|
||||||
moderator: "監察員"
|
moderator: "審查員"
|
||||||
moderation: "監察"
|
moderation: "審查"
|
||||||
nUsersMentioned: "提到了{n}"
|
nUsersMentioned: "提到了{n}"
|
||||||
securityKey: "安全金鑰"
|
securityKey: "安全金鑰"
|
||||||
securityKeyName: "金鑰名稱"
|
securityKeyName: "金鑰名稱"
|
||||||
@@ -421,7 +421,7 @@ invites: "邀請"
|
|||||||
groupName: "群組名稱"
|
groupName: "群組名稱"
|
||||||
members: "成員"
|
members: "成員"
|
||||||
transfer: "轉讓"
|
transfer: "轉讓"
|
||||||
messagingWithUser: "傳送訊息給其他使用者"
|
messagingWithUser: "與其他使用者聊天"
|
||||||
messagingWithGroup: "發送訊息至群組"
|
messagingWithGroup: "發送訊息至群組"
|
||||||
title: "標題"
|
title: "標題"
|
||||||
text: "文字"
|
text: "文字"
|
||||||
@@ -473,7 +473,7 @@ createAccount: "建立帳戶"
|
|||||||
existingAccount: "現有帳戶"
|
existingAccount: "現有帳戶"
|
||||||
regenerate: "再生"
|
regenerate: "再生"
|
||||||
fontSize: "字體大小"
|
fontSize: "字體大小"
|
||||||
noFollowRequests: "沒有要求跟隨您的申請"
|
noFollowRequests: "沒有跟隨您的請求"
|
||||||
openImageInNewTab: "於新分頁中開啟圖片"
|
openImageInNewTab: "於新分頁中開啟圖片"
|
||||||
dashboard: "儀表板"
|
dashboard: "儀表板"
|
||||||
local: "本地"
|
local: "本地"
|
||||||
@@ -530,8 +530,8 @@ installedDate: "安裝時間"
|
|||||||
lastUsedDate: "最後上線日期"
|
lastUsedDate: "最後上線日期"
|
||||||
state: "狀態"
|
state: "狀態"
|
||||||
sort: "排序"
|
sort: "排序"
|
||||||
ascendingOrder: "昇冪"
|
ascendingOrder: "遞增"
|
||||||
descendingOrder: "降冪"
|
descendingOrder: "遞減"
|
||||||
scratchpad: "暫存記憶體"
|
scratchpad: "暫存記憶體"
|
||||||
scratchpadDescription: "AiScript控制台為AiScript提供了實驗環境。您可以在此編寫、執行和確認代碼與Misskey互動的结果。"
|
scratchpadDescription: "AiScript控制台為AiScript提供了實驗環境。您可以在此編寫、執行和確認代碼與Misskey互動的结果。"
|
||||||
output: "輸出"
|
output: "輸出"
|
||||||
@@ -607,7 +607,7 @@ testEmail: "測試郵件發送"
|
|||||||
wordMute: "被靜音的文字"
|
wordMute: "被靜音的文字"
|
||||||
regexpError: "正規表達式錯誤"
|
regexpError: "正規表達式錯誤"
|
||||||
regexpErrorDescription: "{tab} 靜音文字的第 {line} 行的正規表達式有錯誤:"
|
regexpErrorDescription: "{tab} 靜音文字的第 {line} 行的正規表達式有錯誤:"
|
||||||
instanceMute: "實例的靜音"
|
instanceMute: "被靜音的實例"
|
||||||
userSaysSomething: "{name}說了什麼"
|
userSaysSomething: "{name}說了什麼"
|
||||||
makeActive: "啟用"
|
makeActive: "啟用"
|
||||||
display: "檢視"
|
display: "檢視"
|
||||||
@@ -939,6 +939,8 @@ cannotPerformTemporaryDescription: "由於超過操作次數限制,暫時無
|
|||||||
preset: "預設值"
|
preset: "預設值"
|
||||||
selectFromPresets: "從預設值中選擇"
|
selectFromPresets: "從預設值中選擇"
|
||||||
achievements: "成就"
|
achievements: "成就"
|
||||||
|
gotInvalidResponseError: "伺服器的回應無效"
|
||||||
|
gotInvalidResponseErrorDescription: "伺服器可能已關閉或者在維護中,請稍後再試。"
|
||||||
_achievements:
|
_achievements:
|
||||||
earnedAt: "獲得日期"
|
earnedAt: "獲得日期"
|
||||||
_types:
|
_types:
|
||||||
@@ -950,13 +952,13 @@ _achievements:
|
|||||||
title: "若干貼文"
|
title: "若干貼文"
|
||||||
description: "發表了10則貼文"
|
description: "發表了10則貼文"
|
||||||
_notes100:
|
_notes100:
|
||||||
title: "許多的貼文"
|
title: "許多貼文"
|
||||||
description: "發表了100則貼文"
|
description: "發表了100則貼文"
|
||||||
_notes500:
|
_notes500:
|
||||||
title: "滿滿的貼文"
|
title: "滿滿的貼文"
|
||||||
description: "發表了500則貼文"
|
description: "發表了500則貼文"
|
||||||
_notes1000:
|
_notes1000:
|
||||||
title: "一堆貼文"
|
title: "堆積如山的貼文"
|
||||||
description: "發表了1000則貼文"
|
description: "發表了1000則貼文"
|
||||||
_notes5000:
|
_notes5000:
|
||||||
title: "滔滔不絕的貼文"
|
title: "滔滔不絕的貼文"
|
||||||
@@ -995,24 +997,24 @@ _achievements:
|
|||||||
_login3:
|
_login3:
|
||||||
title: "初學者Ⅰ"
|
title: "初學者Ⅰ"
|
||||||
description: "總登入天數為3天"
|
description: "總登入天數為3天"
|
||||||
flavor: "從今天開始,我就是Misskeyist"
|
flavor: "從今天開始,我就是Misskist"
|
||||||
_login7:
|
_login7:
|
||||||
title: "初學者ⅠⅠ"
|
title: "初學者ⅠⅠ"
|
||||||
description: "總登入天數為7天"
|
description: "總登入天數為7天"
|
||||||
flavor: "您開始習慣了嗎?"
|
flavor: "您開始習慣了嗎?"
|
||||||
_login15:
|
_login15:
|
||||||
title: "初學者III"
|
title: "初學者ⅠⅠⅠ"
|
||||||
description: "總登入天數為15天"
|
description: "總登入天數為15天"
|
||||||
_login30:
|
_login30:
|
||||||
title: "Misskeyist Ⅰ"
|
title: "Misskist Ⅰ"
|
||||||
description: "總登入天數為30天"
|
description: "總登入天數為30天"
|
||||||
_login60:
|
_login60:
|
||||||
title: "Misskeyist ⅠⅠ"
|
title: "Misskist ⅠⅠ"
|
||||||
description: "總登入天數為60天"
|
description: "總登入天數為60天"
|
||||||
_login100:
|
_login100:
|
||||||
title: "Misskeyist ⅠⅠⅠ"
|
title: "Misskist ⅠⅠⅠ"
|
||||||
description: "總登入天數為100天"
|
description: "總登入天數為100天"
|
||||||
flavor: "辣個 Misskeyist 用戶"
|
flavor: "辣個 Misskist 用戶"
|
||||||
_login200:
|
_login200:
|
||||||
title: "普通Ⅰ"
|
title: "普通Ⅰ"
|
||||||
description: "總登入天數為200天"
|
description: "總登入天數為200天"
|
||||||
@@ -1042,27 +1044,76 @@ _achievements:
|
|||||||
title: "貼文大師ⅠⅠⅠ"
|
title: "貼文大師ⅠⅠⅠ"
|
||||||
description: "總登入天數為1,000天"
|
description: "總登入天數為1,000天"
|
||||||
flavor: "感謝您使用Misskey!"
|
flavor: "感謝您使用Misskey!"
|
||||||
|
_noteClipped1:
|
||||||
|
title: "忍不住要收進摘錄裡"
|
||||||
|
description: "第一次將貼文收進摘錄"
|
||||||
|
_noteFavorited1:
|
||||||
|
title: "觀星者"
|
||||||
|
description: "第一次將貼文收藏至我的最愛"
|
||||||
|
_myNoteFavorited1:
|
||||||
|
title: "想要星星"
|
||||||
|
description: "自己的貼文被他人收藏至「我的最愛」了"
|
||||||
|
_profileFilled:
|
||||||
|
title: "有備而來"
|
||||||
|
description: "設定了個人檔案"
|
||||||
|
_markedAsCat:
|
||||||
|
title: "我是貓"
|
||||||
|
description: "已將帳戶設定為貓"
|
||||||
|
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: "Follow me!"
|
||||||
|
description: "跟隨者超過10人了"
|
||||||
|
_followers50:
|
||||||
|
title: "成群結隊"
|
||||||
|
description: "跟隨者超過50人了"
|
||||||
|
_followers100:
|
||||||
|
title: "紅人"
|
||||||
|
description: "跟隨者超過100人了"
|
||||||
|
_followers300:
|
||||||
|
title: "請排成一排"
|
||||||
|
description: "跟隨者超過300人了"
|
||||||
_followers500:
|
_followers500:
|
||||||
title: "基站"
|
title: "基地台"
|
||||||
description: "超過500名追隨者"
|
description: "超過500名追隨者了"
|
||||||
_followers1000:
|
_followers1000:
|
||||||
title: "影響者"
|
title: "影響者"
|
||||||
description: "超過1000名追隨者"
|
description: "超過1000名追隨者了"
|
||||||
_collectAchievements30:
|
_collectAchievements30:
|
||||||
title: "成就收藏家"
|
title: "成就收藏家"
|
||||||
description: "獲得30個以上的成就"
|
description: "獲得30個以上的成就"
|
||||||
_viewAchievements3min:
|
_viewAchievements3min:
|
||||||
title: "喜愛成就"
|
title: "喜愛成就"
|
||||||
description: "看成就列表要花了3分鐘以上"
|
description: "看成就列表要花3分鐘以上"
|
||||||
_iLoveMisskey:
|
_iLoveMisskey:
|
||||||
title: "I Love Misskey"
|
title: "I Love Misskey"
|
||||||
description: "發布「I ❤ #Misskey」"
|
description: "發布「I ❤ #Misskey」"
|
||||||
flavor: "感謝您使用Misskey! by 開發團隊"
|
flavor: "感謝您使用Misskey! by 開發團隊"
|
||||||
|
_foundTreasure:
|
||||||
|
title: "尋寶"
|
||||||
|
description: "發現了隱藏的寶藏"
|
||||||
_client30min:
|
_client30min:
|
||||||
title: "休息一下"
|
title: "休息一下"
|
||||||
description: "用戶端啟動已超過30分鐘"
|
description: "用戶端啟動已超過30分鐘"
|
||||||
_noteDeletedWithin1min:
|
_noteDeletedWithin1min:
|
||||||
title: "現在沒有"
|
title: "現在沒有了"
|
||||||
description: "發文後1分鐘內刪文"
|
description: "發文後1分鐘內刪文"
|
||||||
_postedAtLateNight:
|
_postedAtLateNight:
|
||||||
title: "夜行性"
|
title: "夜行性"
|
||||||
@@ -1071,18 +1122,22 @@ _achievements:
|
|||||||
_postedAt0min0sec:
|
_postedAt0min0sec:
|
||||||
title: "報時"
|
title: "報時"
|
||||||
description: "在0分0秒發佈貼文"
|
description: "在0分0秒發佈貼文"
|
||||||
|
flavor: "啵.啵.啵.嗶ー"
|
||||||
_selfQuote:
|
_selfQuote:
|
||||||
title: "自我引用"
|
title: "自我引用"
|
||||||
description: "引用了自己的貼文"
|
description: "引用了自己的貼文"
|
||||||
_htl20npm:
|
_htl20npm:
|
||||||
title: "流動的TL"
|
title: "流動的TL"
|
||||||
description: "在首頁時間軸的流速超過20npm"
|
description: "在首頁時間軸的流速超過20npm"
|
||||||
|
_viewInstanceChart:
|
||||||
|
title: "分析師"
|
||||||
|
description: "顯示了實例的圖表"
|
||||||
_outputHelloWorldOnScratchpad:
|
_outputHelloWorldOnScratchpad:
|
||||||
title: "Hello world!"
|
title: "Hello world!"
|
||||||
description: "在暫存記憶體輸出了 hello world"
|
description: "在暫存記憶體輸出了 hello world"
|
||||||
_open3windows:
|
_open3windows:
|
||||||
title: "多重視窗"
|
title: "多重視窗"
|
||||||
description: "開啟3個以上的視窗"
|
description: "開啟了3個以上的視窗"
|
||||||
_driveFolderCircularReference:
|
_driveFolderCircularReference:
|
||||||
title: "循環引用"
|
title: "循環引用"
|
||||||
description: "試圖遞迴套入雲端硬碟資料夾"
|
description: "試圖遞迴套入雲端硬碟資料夾"
|
||||||
@@ -1098,13 +1153,37 @@ _achievements:
|
|||||||
_setNameToSyuilo:
|
_setNameToSyuilo:
|
||||||
title: "神的情結"
|
title: "神的情結"
|
||||||
description: "將名稱設定為 syuilo"
|
description: "將名稱設定為 syuilo"
|
||||||
|
_passedSinceAccountCreated1:
|
||||||
|
title: "一周年"
|
||||||
|
description: "自建立帳戶開始過了1年"
|
||||||
|
_passedSinceAccountCreated2:
|
||||||
|
title: "二周年"
|
||||||
|
description: "自建立帳戶開始過了2年"
|
||||||
|
_passedSinceAccountCreated3:
|
||||||
|
title: "三周年"
|
||||||
|
description: "自建立帳戶開始過了3年"
|
||||||
|
_loggedInOnBirthday:
|
||||||
|
title: "生日快樂"
|
||||||
|
description: "在生日當天登入了"
|
||||||
|
_loggedInOnNewYearsDay:
|
||||||
|
title: "新年快樂"
|
||||||
|
description: "在元旦當天登入了"
|
||||||
|
flavor: "今年也請對敝實例多多指教"
|
||||||
|
_cookieClicked:
|
||||||
|
title: "點擊餅乾的遊戲"
|
||||||
|
description: "點擊了餅乾"
|
||||||
|
flavor: "是不是軟體有問題?"
|
||||||
|
_brainDiver:
|
||||||
|
title: "Brain Driver"
|
||||||
|
description: "發佈了Brain Driver的連結"
|
||||||
|
flavor: "Misskey-Misskey La-Tu-Ma"
|
||||||
_role:
|
_role:
|
||||||
new: "建立角色"
|
new: "建立角色"
|
||||||
edit: "編輯角色"
|
edit: "編輯角色"
|
||||||
name: "角色名稱"
|
name: "角色名稱"
|
||||||
description: "角色描述 "
|
description: "角色描述 "
|
||||||
permission: "角色的權限"
|
permission: "角色的權限"
|
||||||
descriptionOfPermission: "<b>審核員</b>執行與審核相關的基本操作。\n<b>管理員</b>能變更實例的全部設定。"
|
descriptionOfPermission: "<b>審查員</b>執行與審查相關的基本操作。\n<b>管理員</b>能變更實例的全部設定"
|
||||||
assignTarget: "指派目標"
|
assignTarget: "指派目標"
|
||||||
descriptionOfAssignTarget: "<b>手動</b>是以手動管理這個角色包含的人員。\n<b>符合條件</b>是設定條件以自動包含符合條件的使用者。"
|
descriptionOfAssignTarget: "<b>手動</b>是以手動管理這個角色包含的人員。\n<b>符合條件</b>是設定條件以自動包含符合條件的使用者。"
|
||||||
manual: "手動"
|
manual: "手動"
|
||||||
@@ -1118,8 +1197,11 @@ _role:
|
|||||||
baseRole: "基本角色"
|
baseRole: "基本角色"
|
||||||
useBaseValue: "使用基本角色的值"
|
useBaseValue: "使用基本角色的值"
|
||||||
chooseRoleToAssign: "選擇要指派的角色"
|
chooseRoleToAssign: "選擇要指派的角色"
|
||||||
canEditMembersByModerator: "允許編輯監察員的成員"
|
iconUrl: "圖示的URL"
|
||||||
descriptionOfCanEditMembersByModerator: "如果開啟,管理員與監察員都可以為使用者指派/解除指派該角色。如果關閉,則只有管理員可以執行。"
|
asBadge: "顯示為徽章"
|
||||||
|
descriptionOfAsBadge: "開啟的話,角色圖示會顯示在用戶名旁邊。"
|
||||||
|
canEditMembersByModerator: "允許編輯審查員的成員"
|
||||||
|
descriptionOfCanEditMembersByModerator: "如果開啟,管理員與審查員都可以為使用者指派/解除指派該角色。如果關閉,則只有管理員可以執行。"
|
||||||
priority: "優先級"
|
priority: "優先級"
|
||||||
_priority:
|
_priority:
|
||||||
low: "低"
|
low: "低"
|
||||||
@@ -1156,7 +1238,7 @@ _role:
|
|||||||
or: "~或~"
|
or: "~或~"
|
||||||
not: "~否"
|
not: "~否"
|
||||||
_sensitiveMediaDetection:
|
_sensitiveMediaDetection:
|
||||||
description: "您可以使用機器學習自動檢測敏感媒體並將其用於審核。 伺服器的負荷會稍微增加。"
|
description: "您可以使用機器學習自動檢測敏感媒體並將其用於審查。 伺服器的負荷會稍微增加。"
|
||||||
sensitivity: "檢測敏感度"
|
sensitivity: "檢測敏感度"
|
||||||
sensitivityDescription: "敏感度低時,誤檢測(偽陽性)會減少。敏感度高時,漏檢(偽陰性)會減少。"
|
sensitivityDescription: "敏感度低時,誤檢測(偽陽性)會減少。敏感度高時,漏檢(偽陰性)會減少。"
|
||||||
setSensitiveFlagAutomatically: "設定 NSFW 旗標"
|
setSensitiveFlagAutomatically: "設定 NSFW 旗標"
|
||||||
@@ -1747,6 +1829,7 @@ _notification:
|
|||||||
pollEnded: "問卷調查已產生結果"
|
pollEnded: "問卷調查已產生結果"
|
||||||
unreadAntennaNote: "天線 {name}"
|
unreadAntennaNote: "天線 {name}"
|
||||||
emptyPushNotificationMessage: "推送通知已更新"
|
emptyPushNotificationMessage: "推送通知已更新"
|
||||||
|
achievementEarned: "獲得成就"
|
||||||
_types:
|
_types:
|
||||||
all: "全部 "
|
all: "全部 "
|
||||||
follow: "追隨中"
|
follow: "追隨中"
|
||||||
@@ -1788,5 +1871,6 @@ _deck:
|
|||||||
tl: "時間軸"
|
tl: "時間軸"
|
||||||
antenna: "天線"
|
antenna: "天線"
|
||||||
list: "清單"
|
list: "清單"
|
||||||
|
channel: "頻道"
|
||||||
mentions: "提及"
|
mentions: "提及"
|
||||||
direct: "指定使用者"
|
direct: "指定使用者"
|
||||||
|
26
package.json
26
package.json
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "misskey",
|
"name": "misskey",
|
||||||
"version": "13.1.3",
|
"version": "13.5.5",
|
||||||
"codename": "nasubi",
|
"codename": "nasubi",
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
@@ -19,7 +19,7 @@
|
|||||||
"start": "cd packages/backend && node ./built/boot/index.js",
|
"start": "cd packages/backend && node ./built/boot/index.js",
|
||||||
"start:test": "cd packages/backend && cross-env NODE_ENV=test node ./built/boot/index.js",
|
"start:test": "cd packages/backend && cross-env NODE_ENV=test node ./built/boot/index.js",
|
||||||
"init": "pnpm migrate",
|
"init": "pnpm migrate",
|
||||||
"migrate": "cd packages/backend && pnpm typeorm migration:run -d ormconfig.js",
|
"migrate": "cd packages/backend && pnpm migrate",
|
||||||
"migrateandstart": "pnpm migrate && pnpm start",
|
"migrateandstart": "pnpm migrate && pnpm start",
|
||||||
"gulp": "pnpm exec gulp build",
|
"gulp": "pnpm exec gulp build",
|
||||||
"watch": "pnpm dev",
|
"watch": "pnpm dev",
|
||||||
@@ -28,8 +28,8 @@
|
|||||||
"cy:open": "pnpm cypress open --browser --e2e --config-file=cypress.config.ts",
|
"cy:open": "pnpm cypress open --browser --e2e --config-file=cypress.config.ts",
|
||||||
"cy:run": "pnpm cypress run",
|
"cy:run": "pnpm cypress run",
|
||||||
"e2e": "pnpm start-server-and-test start:test http://localhost:61812 cy: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": "cd packages/backend && pnpm jest",
|
||||||
"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",
|
"jest-and-coverage": "cd packages/backend && pnpm jest-and-coverage",
|
||||||
"test": "pnpm jest",
|
"test": "pnpm jest",
|
||||||
"test-and-coverage": "pnpm jest-and-coverage",
|
"test-and-coverage": "pnpm jest-and-coverage",
|
||||||
"format": "pnpm exec gulp format",
|
"format": "pnpm exec gulp format",
|
||||||
@@ -38,8 +38,8 @@
|
|||||||
"cleanall": "pnpm clean-all"
|
"cleanall": "pnpm clean-all"
|
||||||
},
|
},
|
||||||
"resolutions": {
|
"resolutions": {
|
||||||
"chokidar": "^3.5.3",
|
"chokidar": "3.5.3",
|
||||||
"lodash": "^4.17.21"
|
"lodash": "4.17.21"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"execa": "5.1.1",
|
"execa": "5.1.1",
|
||||||
@@ -49,19 +49,19 @@
|
|||||||
"gulp-replace": "1.1.4",
|
"gulp-replace": "1.1.4",
|
||||||
"gulp-terser": "2.1.0",
|
"gulp-terser": "2.1.0",
|
||||||
"js-yaml": "4.1.0",
|
"js-yaml": "4.1.0",
|
||||||
"typescript": "4.9.4"
|
"typescript": "4.9.5"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@types/gulp": "4.0.10",
|
"@types/gulp": "4.0.10",
|
||||||
"@types/gulp-rename": "2.0.1",
|
"@types/gulp-rename": "2.0.1",
|
||||||
"@typescript-eslint/eslint-plugin": "5.48.2",
|
"@typescript-eslint/eslint-plugin": "5.51.0",
|
||||||
"@typescript-eslint/parser": "5.48.2",
|
"@typescript-eslint/parser": "5.51.0",
|
||||||
"cross-env": "7.0.3",
|
"cross-env": "7.0.3",
|
||||||
"cypress": "12.3.0",
|
"cypress": "12.5.1",
|
||||||
"eslint": "^8.32.0",
|
"eslint": "8.33.0",
|
||||||
"start-server-and-test": "1.15.2"
|
"start-server-and-test": "1.15.3"
|
||||||
},
|
},
|
||||||
"optionalDependencies": {
|
"optionalDependencies": {
|
||||||
"@tensorflow/tfjs-core": "^4.2.0"
|
"@tensorflow/tfjs-core": "4.2.0"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
29
packages/backend/migration/1675404035646-cleanup.js
Normal file
29
packages/backend/migration/1675404035646-cleanup.js
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
export class cleanup1675404035646 {
|
||||||
|
name = 'cleanup1675404035646'
|
||||||
|
|
||||||
|
async up(queryRunner) {
|
||||||
|
await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "enableTwitterIntegration"`);
|
||||||
|
await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "enableGithubIntegration"`);
|
||||||
|
await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "enableDiscordIntegration"`);
|
||||||
|
await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "twitterConsumerKey"`);
|
||||||
|
await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "twitterConsumerSecret"`);
|
||||||
|
await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "githubClientId"`);
|
||||||
|
await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "githubClientSecret"`);
|
||||||
|
await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "discordClientId"`);
|
||||||
|
await queryRunner.query(`ALTER TABLE "meta" DROP COLUMN "discordClientSecret"`);
|
||||||
|
await queryRunner.query(`ALTER TABLE "user_profile" DROP COLUMN "integrations"`);
|
||||||
|
}
|
||||||
|
|
||||||
|
async down(queryRunner) {
|
||||||
|
await queryRunner.query(`ALTER TABLE "user_profile" ADD "integrations" jsonb NOT NULL DEFAULT '{}'`);
|
||||||
|
await queryRunner.query(`ALTER TABLE "meta" ADD "discordClientSecret" character varying(128)`);
|
||||||
|
await queryRunner.query(`ALTER TABLE "meta" ADD "discordClientId" character varying(128)`);
|
||||||
|
await queryRunner.query(`ALTER TABLE "meta" ADD "githubClientSecret" character varying(128)`);
|
||||||
|
await queryRunner.query(`ALTER TABLE "meta" ADD "githubClientId" character varying(128)`);
|
||||||
|
await queryRunner.query(`ALTER TABLE "meta" ADD "twitterConsumerSecret" character varying(128)`);
|
||||||
|
await queryRunner.query(`ALTER TABLE "meta" ADD "twitterConsumerKey" character varying(128)`);
|
||||||
|
await queryRunner.query(`ALTER TABLE "meta" ADD "enableDiscordIntegration" boolean NOT NULL DEFAULT false`);
|
||||||
|
await queryRunner.query(`ALTER TABLE "meta" ADD "enableGithubIntegration" boolean NOT NULL DEFAULT false`);
|
||||||
|
await queryRunner.query(`ALTER TABLE "meta" ADD "enableTwitterIntegration" boolean NOT NULL DEFAULT false`);
|
||||||
|
}
|
||||||
|
}
|
13
packages/backend/migration/1675557528704-role-icon-badge.js
Normal file
13
packages/backend/migration/1675557528704-role-icon-badge.js
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
export class roleIconBadge1675557528704 {
|
||||||
|
name = 'roleIconBadge1675557528704'
|
||||||
|
|
||||||
|
async up(queryRunner) {
|
||||||
|
await queryRunner.query(`ALTER TABLE "role" ADD "iconUrl" character varying(512)`);
|
||||||
|
await queryRunner.query(`ALTER TABLE "role" ADD "asBadge" boolean NOT NULL DEFAULT false`);
|
||||||
|
}
|
||||||
|
|
||||||
|
async down(queryRunner) {
|
||||||
|
await queryRunner.query(`ALTER TABLE "role" DROP COLUMN "asBadge"`);
|
||||||
|
await queryRunner.query(`ALTER TABLE "role" DROP COLUMN "iconUrl"`);
|
||||||
|
}
|
||||||
|
}
|
@@ -1,6 +1,6 @@
|
|||||||
import { DataSource } from 'typeorm';
|
import { DataSource } from 'typeorm';
|
||||||
import { loadConfig } from './built/config.js';
|
import { loadConfig } from './built/config.js';
|
||||||
import { entities } from './built/postgre.js';
|
import { entities } from './built/postgres.js';
|
||||||
|
|
||||||
const config = loadConfig();
|
const config = loadConfig();
|
||||||
|
|
||||||
|
@@ -19,34 +19,34 @@
|
|||||||
"test-and-coverage": "pnpm jest-and-coverage"
|
"test-and-coverage": "pnpm jest-and-coverage"
|
||||||
},
|
},
|
||||||
"optionalDependencies": {
|
"optionalDependencies": {
|
||||||
"@tensorflow/tfjs": "^4.1.0",
|
"@tensorflow/tfjs": "4.2.0",
|
||||||
"@tensorflow/tfjs-node": "4.1.0"
|
"@tensorflow/tfjs-node": "4.2.0"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@bull-board/api": "^4.10.2",
|
"@bull-board/api": "4.11.1",
|
||||||
"@bull-board/fastify": "^4.10.2",
|
"@bull-board/fastify": "4.11.1",
|
||||||
"@bull-board/ui": "^4.10.2",
|
"@bull-board/ui": "4.11.1",
|
||||||
"@discordapp/twemoji": "14.0.2",
|
"@discordapp/twemoji": "14.0.2",
|
||||||
"@fastify/accepts": "4.1.0",
|
"@fastify/accepts": "4.1.0",
|
||||||
"@fastify/cookie": "^8.3.0",
|
"@fastify/cookie": "8.3.0",
|
||||||
"@fastify/cors": "8.2.0",
|
"@fastify/cors": "8.2.0",
|
||||||
"@fastify/http-proxy": "^8.4.0",
|
"@fastify/http-proxy": "8.4.0",
|
||||||
"@fastify/multipart": "7.4.0",
|
"@fastify/multipart": "7.4.0",
|
||||||
"@fastify/static": "6.6.1",
|
"@fastify/static": "6.8.0",
|
||||||
"@fastify/view": "7.4.0",
|
"@fastify/view": "7.4.1",
|
||||||
"@nestjs/common": "9.2.1",
|
"@nestjs/common": "9.3.7",
|
||||||
"@nestjs/core": "9.2.1",
|
"@nestjs/core": "9.3.7",
|
||||||
"@nestjs/testing": "9.2.1",
|
"@nestjs/testing": "9.3.7",
|
||||||
"@peertube/http-signature": "1.7.0",
|
"@peertube/http-signature": "1.7.0",
|
||||||
"@sinonjs/fake-timers": "10.0.2",
|
"@sinonjs/fake-timers": "10.0.2",
|
||||||
"accepts": "^1.3.8",
|
"accepts": "1.3.8",
|
||||||
"ajv": "8.12.0",
|
"ajv": "8.12.0",
|
||||||
"archiver": "5.3.1",
|
"archiver": "5.3.1",
|
||||||
"autwh": "0.1.0",
|
"autwh": "0.1.0",
|
||||||
"aws-sdk": "2.1295.0",
|
"aws-sdk": "2.1295.0",
|
||||||
"bcryptjs": "2.4.3",
|
"bcryptjs": "2.4.3",
|
||||||
"blurhash": "2.0.4",
|
"blurhash": "2.0.4",
|
||||||
"bull": "4.10.2",
|
"bull": "4.10.3",
|
||||||
"cacheable-lookup": "6.1.0",
|
"cacheable-lookup": "6.1.0",
|
||||||
"cbor": "8.1.0",
|
"cbor": "8.1.0",
|
||||||
"chalk": "5.2.0",
|
"chalk": "5.2.0",
|
||||||
@@ -62,35 +62,35 @@
|
|||||||
"feed": "4.2.2",
|
"feed": "4.2.2",
|
||||||
"file-type": "18.2.0",
|
"file-type": "18.2.0",
|
||||||
"fluent-ffmpeg": "2.1.2",
|
"fluent-ffmpeg": "2.1.2",
|
||||||
"form-data": "^4.0.0",
|
"form-data": "4.0.0",
|
||||||
"got": "12.5.3",
|
"got": "12.5.3",
|
||||||
"hpagent": "1.2.0",
|
"hpagent": "1.2.0",
|
||||||
"ioredis": "4.28.5",
|
"ioredis": "4.28.5",
|
||||||
"ip-cidr": "3.0.11",
|
"ip-cidr": "3.1.0",
|
||||||
"is-svg": "4.3.2",
|
"is-svg": "4.3.2",
|
||||||
"js-yaml": "4.1.0",
|
"js-yaml": "4.1.0",
|
||||||
"jsdom": "21.0.0",
|
"jsdom": "21.1.0",
|
||||||
"json5": "2.2.3",
|
"json5": "2.2.3",
|
||||||
"json5-loader": "4.0.1",
|
|
||||||
"jsonld": "8.1.0",
|
"jsonld": "8.1.0",
|
||||||
"jsrsasign": "10.6.1",
|
"jsrsasign": "10.6.1",
|
||||||
"mfm-js": "0.23.3",
|
"mfm-js": "0.23.3",
|
||||||
"mime-types": "2.1.35",
|
"mime-types": "2.1.35",
|
||||||
"misskey-js": "0.0.14",
|
"misskey-js": "0.0.15",
|
||||||
"ms": "3.0.0-canary.1",
|
"ms": "3.0.0-canary.1",
|
||||||
"nested-property": "4.0.0",
|
"nested-property": "4.0.0",
|
||||||
"nodemailer": "6.9.0",
|
"node-fetch": "3.3.0",
|
||||||
|
"nodemailer": "6.9.1",
|
||||||
"nsfwjs": "2.4.2",
|
"nsfwjs": "2.4.2",
|
||||||
"oauth": "^0.10.0",
|
"oauth": "0.10.0",
|
||||||
"os-utils": "0.0.14",
|
"os-utils": "0.0.14",
|
||||||
"parse5": "7.1.2",
|
"parse5": "7.1.2",
|
||||||
"pg": "8.8.0",
|
"pg": "8.9.0",
|
||||||
"private-ip": "3.0.0",
|
"private-ip": "3.0.0",
|
||||||
"probe-image-size": "7.2.3",
|
"probe-image-size": "7.2.3",
|
||||||
"promise-limit": "2.7.0",
|
"promise-limit": "2.7.0",
|
||||||
"pug": "3.0.2",
|
"pug": "3.0.2",
|
||||||
"punycode": "2.3.0",
|
"punycode": "2.3.0",
|
||||||
"pureimage": "0.3.15",
|
"pureimage": "0.3.17",
|
||||||
"qrcode": "1.5.1",
|
"qrcode": "1.5.1",
|
||||||
"random-seed": "0.3.0",
|
"random-seed": "0.3.0",
|
||||||
"ratelimiter": "3.4.1",
|
"ratelimiter": "3.4.1",
|
||||||
@@ -102,25 +102,23 @@
|
|||||||
"rss-parser": "3.12.0",
|
"rss-parser": "3.12.0",
|
||||||
"rxjs": "7.8.0",
|
"rxjs": "7.8.0",
|
||||||
"s-age": "1.1.2",
|
"s-age": "1.1.2",
|
||||||
"sanitize-html": "2.8.1",
|
"sanitize-html": "2.9.0",
|
||||||
"seedrandom": "^3.0.5",
|
"seedrandom": "3.0.5",
|
||||||
"semver": "7.3.8",
|
"semver": "7.3.8",
|
||||||
"sharp": "0.31.3",
|
"sharp": "0.31.3",
|
||||||
"speakeasy": "2.0.0",
|
"speakeasy": "2.0.0",
|
||||||
"strict-event-emitter-types": "2.0.0",
|
"strict-event-emitter-types": "2.0.0",
|
||||||
"stringz": "2.1.0",
|
"stringz": "2.1.0",
|
||||||
"summaly": "2.7.0",
|
"summaly": "2.7.0",
|
||||||
"syslog-pro": "git+https://github.com/misskey-dev/SyslogPro#0.2.9-misskey.2",
|
"systeminformation": "5.17.8",
|
||||||
"systeminformation": "5.17.3",
|
"tinycolor2": "1.6.0",
|
||||||
"tinycolor2": "1.5.2",
|
|
||||||
"tmp": "0.2.1",
|
"tmp": "0.2.1",
|
||||||
"tsc-alias": "1.8.2",
|
"tsc-alias": "1.8.2",
|
||||||
"tsconfig-paths": "4.1.2",
|
"tsconfig-paths": "4.1.2",
|
||||||
"twemoji-parser": "14.0.0",
|
"twemoji-parser": "14.0.0",
|
||||||
"typeorm": "0.3.11",
|
"typeorm": "0.3.12",
|
||||||
"typescript": "4.9.4",
|
"typescript": "4.9.5",
|
||||||
"ulid": "2.3.0",
|
"ulid": "2.3.0",
|
||||||
"undici": "^5.15.1",
|
|
||||||
"unzipper": "0.10.11",
|
"unzipper": "0.10.11",
|
||||||
"uuid": "9.0.0",
|
"uuid": "9.0.0",
|
||||||
"vary": "1.1.2",
|
"vary": "1.1.2",
|
||||||
@@ -130,27 +128,28 @@
|
|||||||
"xev": "3.0.2"
|
"xev": "3.0.2"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@redocly/openapi-core": "1.0.0-beta.120",
|
"@jest/globals": "29.4.2",
|
||||||
"@swc/cli": "^0.1.59",
|
"@redocly/openapi-core": "1.0.0-beta.123",
|
||||||
"@swc/core": "1.3.27",
|
"@swc/cli": "0.1.61",
|
||||||
|
"@swc/core": "1.3.34",
|
||||||
"@swc/jest": "0.2.24",
|
"@swc/jest": "0.2.24",
|
||||||
"@types/accepts": "1.3.5",
|
"@types/accepts": "1.3.5",
|
||||||
"@types/archiver": "5.3.1",
|
"@types/archiver": "5.3.1",
|
||||||
"@types/bcryptjs": "2.4.2",
|
"@types/bcryptjs": "2.4.2",
|
||||||
"@types/bull": "4.10.0",
|
"@types/bull": "4.10.0",
|
||||||
"@types/cbor": "6.0.0",
|
"@types/cbor": "6.0.0",
|
||||||
"@types/color-convert": "^2.0.0",
|
"@types/color-convert": "2.0.0",
|
||||||
"@types/content-disposition": "^0.5.5",
|
"@types/content-disposition": "0.5.5",
|
||||||
"@types/escape-regexp": "0.0.1",
|
"@types/escape-regexp": "0.0.1",
|
||||||
"@types/fluent-ffmpeg": "2.1.20",
|
"@types/fluent-ffmpeg": "2.1.20",
|
||||||
"@types/ioredis": "4.28.10",
|
"@types/ioredis": "4.28.10",
|
||||||
"@types/jest": "29.2.6",
|
"@types/jest": "29.4.0",
|
||||||
"@types/js-yaml": "4.0.5",
|
"@types/js-yaml": "4.0.5",
|
||||||
"@types/jsdom": "20.0.1",
|
"@types/jsdom": "21.1.0",
|
||||||
"@types/jsonld": "1.5.8",
|
"@types/jsonld": "1.5.8",
|
||||||
"@types/jsrsasign": "10.5.4",
|
"@types/jsrsasign": "10.5.5",
|
||||||
"@types/mime-types": "2.1.1",
|
"@types/mime-types": "2.1.1",
|
||||||
"@types/node": "18.11.18",
|
"@types/node": "18.13.0",
|
||||||
"@types/node-fetch": "3.0.3",
|
"@types/node-fetch": "3.0.3",
|
||||||
"@types/nodemailer": "6.4.7",
|
"@types/nodemailer": "6.4.7",
|
||||||
"@types/oauth": "0.9.1",
|
"@types/oauth": "0.9.1",
|
||||||
@@ -167,7 +166,6 @@
|
|||||||
"@types/sharp": "0.31.1",
|
"@types/sharp": "0.31.1",
|
||||||
"@types/sinonjs__fake-timers": "8.1.2",
|
"@types/sinonjs__fake-timers": "8.1.2",
|
||||||
"@types/speakeasy": "2.0.7",
|
"@types/speakeasy": "2.0.7",
|
||||||
"@types/syslog-pro": "^1.0.0",
|
|
||||||
"@types/tinycolor2": "1.4.3",
|
"@types/tinycolor2": "1.4.3",
|
||||||
"@types/tmp": "0.2.3",
|
"@types/tmp": "0.2.3",
|
||||||
"@types/unzipper": "0.10.5",
|
"@types/unzipper": "0.10.5",
|
||||||
@@ -176,14 +174,13 @@
|
|||||||
"@types/web-push": "3.3.2",
|
"@types/web-push": "3.3.2",
|
||||||
"@types/websocket": "1.0.5",
|
"@types/websocket": "1.0.5",
|
||||||
"@types/ws": "8.5.4",
|
"@types/ws": "8.5.4",
|
||||||
"@typescript-eslint/eslint-plugin": "5.48.2",
|
"@typescript-eslint/eslint-plugin": "5.51.0",
|
||||||
"@typescript-eslint/parser": "5.48.2",
|
"@typescript-eslint/parser": "5.51.0",
|
||||||
"cross-env": "7.0.3",
|
"cross-env": "7.0.3",
|
||||||
"eslint": "8.32.0",
|
"eslint": "8.33.0",
|
||||||
"eslint-plugin-import": "2.27.5",
|
"eslint-plugin-import": "2.27.5",
|
||||||
"execa": "6.1.0",
|
"execa": "6.1.0",
|
||||||
"jest": "29.3.1",
|
"jest": "29.4.2",
|
||||||
"jest-mock": "^29.3.1",
|
"jest-mock": "29.4.2"
|
||||||
"node-fetch": "3.3.0"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -4,7 +4,7 @@ import { DataSource } from 'typeorm';
|
|||||||
import { createRedisConnection } from '@/redis.js';
|
import { createRedisConnection } from '@/redis.js';
|
||||||
import { DI } from './di-symbols.js';
|
import { DI } from './di-symbols.js';
|
||||||
import { loadConfig } from './config.js';
|
import { loadConfig } from './config.js';
|
||||||
import { createPostgreDataSource } from './postgre.js';
|
import { createPostgresDataSource } from './postgres.js';
|
||||||
import { RepositoryModule } from './models/RepositoryModule.js';
|
import { RepositoryModule } from './models/RepositoryModule.js';
|
||||||
import type { Provider, OnApplicationShutdown } from '@nestjs/common';
|
import type { Provider, OnApplicationShutdown } from '@nestjs/common';
|
||||||
|
|
||||||
@@ -18,7 +18,7 @@ const $config: Provider = {
|
|||||||
const $db: Provider = {
|
const $db: Provider = {
|
||||||
provide: DI.db,
|
provide: DI.db,
|
||||||
useFactory: async (config) => {
|
useFactory: async (config) => {
|
||||||
const db = createPostgreDataSource(config);
|
const db = createPostgresDataSource(config);
|
||||||
return await db.initialize();
|
return await db.initialize();
|
||||||
},
|
},
|
||||||
inject: [DI.config],
|
inject: [DI.config],
|
||||||
|
35
packages/backend/src/boot/common.ts
Normal file
35
packages/backend/src/boot/common.ts
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
import { NestFactory } from '@nestjs/core';
|
||||||
|
import { ChartManagementService } from '@/core/chart/ChartManagementService.js';
|
||||||
|
import { QueueProcessorService } from '@/queue/QueueProcessorService.js';
|
||||||
|
import { NestLogger } from '@/NestLogger.js';
|
||||||
|
import { QueueProcessorModule } from '@/queue/QueueProcessorModule.js';
|
||||||
|
import { JanitorService } from '@/daemons/JanitorService.js';
|
||||||
|
import { QueueStatsService } from '@/daemons/QueueStatsService.js';
|
||||||
|
import { ServerStatsService } from '@/daemons/ServerStatsService.js';
|
||||||
|
import { ServerService } from '@/server/ServerService.js';
|
||||||
|
import { MainModule } from '@/MainModule.js';
|
||||||
|
|
||||||
|
export async function server() {
|
||||||
|
const app = await NestFactory.createApplicationContext(MainModule, {
|
||||||
|
logger: new NestLogger(),
|
||||||
|
});
|
||||||
|
app.enableShutdownHooks();
|
||||||
|
|
||||||
|
const serverService = app.get(ServerService);
|
||||||
|
serverService.launch();
|
||||||
|
|
||||||
|
app.get(ChartManagementService).start();
|
||||||
|
app.get(JanitorService).start();
|
||||||
|
app.get(QueueStatsService).start();
|
||||||
|
app.get(ServerStatsService).start();
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function jobQueue() {
|
||||||
|
const jobQueue = await NestFactory.createApplicationContext(QueueProcessorModule, {
|
||||||
|
logger: new NestLogger(),
|
||||||
|
});
|
||||||
|
jobQueue.enableShutdownHooks();
|
||||||
|
|
||||||
|
jobQueue.get(QueueProcessorService).start();
|
||||||
|
jobQueue.get(ChartManagementService).start();
|
||||||
|
}
|
@@ -6,21 +6,12 @@ import cluster from 'node:cluster';
|
|||||||
import chalk from 'chalk';
|
import chalk from 'chalk';
|
||||||
import chalkTemplate from 'chalk-template';
|
import chalkTemplate from 'chalk-template';
|
||||||
import semver from 'semver';
|
import semver from 'semver';
|
||||||
import { NestFactory } from '@nestjs/core';
|
|
||||||
import Logger from '@/logger.js';
|
import Logger from '@/logger.js';
|
||||||
import { loadConfig } from '@/config.js';
|
import { loadConfig } from '@/config.js';
|
||||||
import type { Config } from '@/config.js';
|
import type { Config } from '@/config.js';
|
||||||
import { lessThan } from '@/misc/prelude/array.js';
|
|
||||||
import { showMachineInfo } from '@/misc/show-machine-info.js';
|
import { showMachineInfo } from '@/misc/show-machine-info.js';
|
||||||
import { DaemonModule } from '@/daemons/DaemonModule.js';
|
import { envOption } from '@/env.js';
|
||||||
import { JanitorService } from '@/daemons/JanitorService.js';
|
import { jobQueue, server } from './common.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);
|
const _filename = fileURLToPath(import.meta.url);
|
||||||
const _dirname = dirname(_filename);
|
const _dirname = dirname(_filename);
|
||||||
@@ -73,14 +64,13 @@ export async function masterMain() {
|
|||||||
process.exit(1);
|
process.exit(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
const app = await NestFactory.createApplicationContext(MainModule, {
|
if (envOption.onlyServer) {
|
||||||
logger: new NestLogger(),
|
await server();
|
||||||
});
|
} else if (envOption.onlyQueue) {
|
||||||
app.enableShutdownHooks();
|
await jobQueue();
|
||||||
|
} else {
|
||||||
// start server
|
await server();
|
||||||
const serverService = app.get(ServerService);
|
}
|
||||||
serverService.launch();
|
|
||||||
|
|
||||||
bootLogger.succ('Misskey initialized');
|
bootLogger.succ('Misskey initialized');
|
||||||
|
|
||||||
@@ -89,11 +79,6 @@ export async function masterMain() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
bootLogger.succ(`Now listening on port ${config.port} on ${config.url}`, null, true);
|
bootLogger.succ(`Now listening on port ${config.port} on ${config.url}`, null, true);
|
||||||
|
|
||||||
app.get(ChartManagementService).start();
|
|
||||||
app.get(JanitorService).start();
|
|
||||||
app.get(QueueStatsService).start();
|
|
||||||
app.get(ServerStatsService).start();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function showEnvironment(): void {
|
function showEnvironment(): void {
|
||||||
|
@@ -1,23 +1,18 @@
|
|||||||
import cluster from 'node:cluster';
|
import cluster from 'node:cluster';
|
||||||
import { NestFactory } from '@nestjs/core';
|
import { envOption } from '@/env.js';
|
||||||
import { ChartManagementService } from '@/core/chart/ChartManagementService.js';
|
import { jobQueue, server } from './common.js';
|
||||||
import { QueueProcessorService } from '@/queue/QueueProcessorService.js';
|
|
||||||
import { NestLogger } from '@/NestLogger.js';
|
|
||||||
import { QueueProcessorModule } from '@/queue/QueueProcessorModule.js';
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Init worker process
|
* Init worker process
|
||||||
*/
|
*/
|
||||||
export async function workerMain() {
|
export async function workerMain() {
|
||||||
const jobQueue = await NestFactory.createApplicationContext(QueueProcessorModule, {
|
if (envOption.onlyServer) {
|
||||||
logger: new NestLogger(),
|
await server();
|
||||||
});
|
} else if (envOption.onlyQueue) {
|
||||||
jobQueue.enableShutdownHooks();
|
await jobQueue();
|
||||||
|
} else {
|
||||||
// start job queue
|
await jobQueue();
|
||||||
jobQueue.get(QueueProcessorService).start();
|
}
|
||||||
|
|
||||||
jobQueue.get(ChartManagementService).start();
|
|
||||||
|
|
||||||
if (cluster.isWorker) {
|
if (cluster.isWorker) {
|
||||||
// Send a 'ready' message to parent process
|
// Send a 'ready' message to parent process
|
||||||
|
@@ -65,11 +65,6 @@ export type Source = {
|
|||||||
deliverJobMaxAttempts?: number;
|
deliverJobMaxAttempts?: number;
|
||||||
inboxJobMaxAttempts?: number;
|
inboxJobMaxAttempts?: number;
|
||||||
|
|
||||||
syslog: {
|
|
||||||
host: string;
|
|
||||||
port: number;
|
|
||||||
};
|
|
||||||
|
|
||||||
mediaProxy?: string;
|
mediaProxy?: string;
|
||||||
proxyRemoteFiles?: boolean;
|
proxyRemoteFiles?: boolean;
|
||||||
|
|
||||||
@@ -92,6 +87,8 @@ export type Mixin = {
|
|||||||
userAgent: string;
|
userAgent: string;
|
||||||
clientEntry: string;
|
clientEntry: string;
|
||||||
clientManifestExists: boolean;
|
clientManifestExists: boolean;
|
||||||
|
mediaProxy: string;
|
||||||
|
externalMediaProxyEnabled: boolean;
|
||||||
};
|
};
|
||||||
|
|
||||||
export type Config = Source & Mixin;
|
export type Config = Source & Mixin;
|
||||||
@@ -113,7 +110,7 @@ const path = process.env.NODE_ENV === 'test'
|
|||||||
|
|
||||||
export function loadConfig() {
|
export function loadConfig() {
|
||||||
const meta = JSON.parse(fs.readFileSync(`${_dirname}/../../../built/meta.json`, 'utf-8'));
|
const meta = JSON.parse(fs.readFileSync(`${_dirname}/../../../built/meta.json`, 'utf-8'));
|
||||||
const clientManifestExists = fs.existsSync(_dirname + '/../../../built/_vite_/manifest.json')
|
const clientManifestExists = fs.existsSync(_dirname + '/../../../built/_vite_/manifest.json');
|
||||||
const clientManifest = clientManifestExists ?
|
const clientManifest = clientManifestExists ?
|
||||||
JSON.parse(fs.readFileSync(`${_dirname}/../../../built/_vite_/manifest.json`, 'utf-8'))
|
JSON.parse(fs.readFileSync(`${_dirname}/../../../built/_vite_/manifest.json`, 'utf-8'))
|
||||||
: { 'src/init.ts': { file: 'src/init.ts' } };
|
: { 'src/init.ts': { file: 'src/init.ts' } };
|
||||||
@@ -140,6 +137,13 @@ export function loadConfig() {
|
|||||||
mixin.clientEntry = clientManifest['src/init.ts'];
|
mixin.clientEntry = clientManifest['src/init.ts'];
|
||||||
mixin.clientManifestExists = clientManifestExists;
|
mixin.clientManifestExists = clientManifestExists;
|
||||||
|
|
||||||
|
const externalMediaProxy = config.mediaProxy ?
|
||||||
|
config.mediaProxy.endsWith('/') ? config.mediaProxy.substring(0, config.mediaProxy.length - 1) : config.mediaProxy
|
||||||
|
: null;
|
||||||
|
const internalMediaProxy = `${mixin.scheme}://${mixin.host}/proxy`;
|
||||||
|
mixin.mediaProxy = externalMediaProxy ?? internalMediaProxy;
|
||||||
|
mixin.externalMediaProxyEnabled = externalMediaProxy !== null && externalMediaProxy !== internalMediaProxy;
|
||||||
|
|
||||||
if (!config.redis.prefix) config.redis.prefix = mixin.host;
|
if (!config.redis.prefix) config.redis.prefix = mixin.host;
|
||||||
|
|
||||||
return Object.assign(config, mixin);
|
return Object.assign(config, mixin);
|
||||||
|
@@ -5,7 +5,7 @@ import { DI } from '@/di-symbols.js';
|
|||||||
import { bindThis } from '@/decorators.js';
|
import { bindThis } from '@/decorators.js';
|
||||||
import { CreateNotificationService } from '@/core/CreateNotificationService.js';
|
import { CreateNotificationService } from '@/core/CreateNotificationService.js';
|
||||||
|
|
||||||
const ACHIEVEMENT_TYPES = [
|
export const ACHIEVEMENT_TYPES = [
|
||||||
'notes1',
|
'notes1',
|
||||||
'notes10',
|
'notes10',
|
||||||
'notes100',
|
'notes100',
|
||||||
@@ -44,6 +44,7 @@ const ACHIEVEMENT_TYPES = [
|
|||||||
'loggedInOnNewYearsDay',
|
'loggedInOnNewYearsDay',
|
||||||
'noteClipped1',
|
'noteClipped1',
|
||||||
'noteFavorited1',
|
'noteFavorited1',
|
||||||
|
'myNoteFavorited1',
|
||||||
'profileFilled',
|
'profileFilled',
|
||||||
'markedAsCat',
|
'markedAsCat',
|
||||||
'following1',
|
'following1',
|
||||||
@@ -61,12 +62,14 @@ const ACHIEVEMENT_TYPES = [
|
|||||||
'collectAchievements30',
|
'collectAchievements30',
|
||||||
'viewAchievements3min',
|
'viewAchievements3min',
|
||||||
'iLoveMisskey',
|
'iLoveMisskey',
|
||||||
|
'foundTreasure',
|
||||||
'client30min',
|
'client30min',
|
||||||
'noteDeletedWithin1min',
|
'noteDeletedWithin1min',
|
||||||
'postedAtLateNight',
|
'postedAtLateNight',
|
||||||
'postedAt0min0sec',
|
'postedAt0min0sec',
|
||||||
'selfQuote',
|
'selfQuote',
|
||||||
'htl20npm',
|
'htl20npm',
|
||||||
|
'viewInstanceChart',
|
||||||
'outputHelloWorldOnScratchpad',
|
'outputHelloWorldOnScratchpad',
|
||||||
'open3windows',
|
'open3windows',
|
||||||
'driveFolderCircularReference',
|
'driveFolderCircularReference',
|
||||||
@@ -94,7 +97,7 @@ export class AchievementService {
|
|||||||
@bindThis
|
@bindThis
|
||||||
public async create(
|
public async create(
|
||||||
userId: User['id'],
|
userId: User['id'],
|
||||||
type: string,
|
type: typeof ACHIEVEMENT_TYPES[number],
|
||||||
): Promise<void> {
|
): Promise<void> {
|
||||||
if (!ACHIEVEMENT_TYPES.includes(type)) return;
|
if (!ACHIEVEMENT_TYPES.includes(type)) return;
|
||||||
|
|
||||||
|
@@ -10,10 +10,9 @@ import { isUserRelated } from '@/misc/is-user-related.js';
|
|||||||
import { GlobalEventService } from '@/core/GlobalEventService.js';
|
import { GlobalEventService } from '@/core/GlobalEventService.js';
|
||||||
import { PushNotificationService } from '@/core/PushNotificationService.js';
|
import { PushNotificationService } from '@/core/PushNotificationService.js';
|
||||||
import * as Acct from '@/misc/acct.js';
|
import * as Acct from '@/misc/acct.js';
|
||||||
import { Cache } from '@/misc/cache.js';
|
|
||||||
import type { Packed } from '@/misc/schema.js';
|
import type { Packed } from '@/misc/schema.js';
|
||||||
import { DI } from '@/di-symbols.js';
|
import { DI } from '@/di-symbols.js';
|
||||||
import type { MutingsRepository, BlockingsRepository, NotesRepository, AntennaNotesRepository, AntennasRepository, UserGroupJoiningsRepository, UserListJoiningsRepository } from '@/models/index.js';
|
import type { MutingsRepository, NotesRepository, AntennaNotesRepository, AntennasRepository, UserGroupJoiningsRepository, UserListJoiningsRepository } from '@/models/index.js';
|
||||||
import { UtilityService } from '@/core/UtilityService.js';
|
import { UtilityService } from '@/core/UtilityService.js';
|
||||||
import { bindThis } from '@/decorators.js';
|
import { bindThis } from '@/decorators.js';
|
||||||
import { StreamMessages } from '@/server/api/stream/types.js';
|
import { StreamMessages } from '@/server/api/stream/types.js';
|
||||||
@@ -23,7 +22,6 @@ import type { OnApplicationShutdown } from '@nestjs/common';
|
|||||||
export class AntennaService implements OnApplicationShutdown {
|
export class AntennaService implements OnApplicationShutdown {
|
||||||
private antennasFetched: boolean;
|
private antennasFetched: boolean;
|
||||||
private antennas: Antenna[];
|
private antennas: Antenna[];
|
||||||
private blockingCache: Cache<User['id'][]>;
|
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
@Inject(DI.redisSubscriber)
|
@Inject(DI.redisSubscriber)
|
||||||
@@ -32,9 +30,6 @@ export class AntennaService implements OnApplicationShutdown {
|
|||||||
@Inject(DI.mutingsRepository)
|
@Inject(DI.mutingsRepository)
|
||||||
private mutingsRepository: MutingsRepository,
|
private mutingsRepository: MutingsRepository,
|
||||||
|
|
||||||
@Inject(DI.blockingsRepository)
|
|
||||||
private blockingsRepository: BlockingsRepository,
|
|
||||||
|
|
||||||
@Inject(DI.notesRepository)
|
@Inject(DI.notesRepository)
|
||||||
private notesRepository: NotesRepository,
|
private notesRepository: NotesRepository,
|
||||||
|
|
||||||
@@ -52,14 +47,13 @@ export class AntennaService implements OnApplicationShutdown {
|
|||||||
|
|
||||||
private utilityService: UtilityService,
|
private utilityService: UtilityService,
|
||||||
private idService: IdService,
|
private idService: IdService,
|
||||||
private globalEventServie: GlobalEventService,
|
private globalEventService: GlobalEventService,
|
||||||
private pushNotificationService: PushNotificationService,
|
private pushNotificationService: PushNotificationService,
|
||||||
private noteEntityService: NoteEntityService,
|
private noteEntityService: NoteEntityService,
|
||||||
private antennaEntityService: AntennaEntityService,
|
private antennaEntityService: AntennaEntityService,
|
||||||
) {
|
) {
|
||||||
this.antennasFetched = false;
|
this.antennasFetched = false;
|
||||||
this.antennas = [];
|
this.antennas = [];
|
||||||
this.blockingCache = new Cache<User['id'][]>(1000 * 60 * 5);
|
|
||||||
|
|
||||||
this.redisSubscriber.on('message', this.onRedisMessage);
|
this.redisSubscriber.on('message', this.onRedisMessage);
|
||||||
}
|
}
|
||||||
@@ -77,10 +71,16 @@ export class AntennaService implements OnApplicationShutdown {
|
|||||||
const { type, body } = obj.message as StreamMessages['internal']['payload'];
|
const { type, body } = obj.message as StreamMessages['internal']['payload'];
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case 'antennaCreated':
|
case 'antennaCreated':
|
||||||
this.antennas.push(body);
|
this.antennas.push({
|
||||||
|
...body,
|
||||||
|
createdAt: new Date(body.createdAt),
|
||||||
|
});
|
||||||
break;
|
break;
|
||||||
case 'antennaUpdated':
|
case 'antennaUpdated':
|
||||||
this.antennas[this.antennas.findIndex(a => a.id === body.id)] = body;
|
this.antennas[this.antennas.findIndex(a => a.id === body.id)] = {
|
||||||
|
...body,
|
||||||
|
createdAt: new Date(body.createdAt),
|
||||||
|
};
|
||||||
break;
|
break;
|
||||||
case 'antennaDeleted':
|
case 'antennaDeleted':
|
||||||
this.antennas = this.antennas.filter(a => a.id !== body.id);
|
this.antennas = this.antennas.filter(a => a.id !== body.id);
|
||||||
@@ -103,7 +103,7 @@ export class AntennaService implements OnApplicationShutdown {
|
|||||||
read: read,
|
read: read,
|
||||||
});
|
});
|
||||||
|
|
||||||
this.globalEventServie.publishAntennaStream(antenna.id, 'note', note);
|
this.globalEventService.publishAntennaStream(antenna.id, 'note', note);
|
||||||
|
|
||||||
if (!read) {
|
if (!read) {
|
||||||
const mutings = await this.mutingsRepository.find({
|
const mutings = await this.mutingsRepository.find({
|
||||||
@@ -133,7 +133,7 @@ export class AntennaService implements OnApplicationShutdown {
|
|||||||
setTimeout(async () => {
|
setTimeout(async () => {
|
||||||
const unread = await this.antennaNotesRepository.findOneBy({ antennaId: antenna.id, read: false });
|
const unread = await this.antennaNotesRepository.findOneBy({ antennaId: antenna.id, read: false });
|
||||||
if (unread) {
|
if (unread) {
|
||||||
this.globalEventServie.publishMainStream(antenna.userId, 'unreadAntenna', antenna);
|
this.globalEventService.publishMainStream(antenna.userId, 'unreadAntenna', antenna);
|
||||||
this.pushNotificationService.pushNotification(antenna.userId, 'unreadAntennaNote', {
|
this.pushNotificationService.pushNotification(antenna.userId, 'unreadAntennaNote', {
|
||||||
antenna: { id: antenna.id, name: antenna.name },
|
antenna: { id: antenna.id, name: antenna.name },
|
||||||
note: await this.noteEntityService.pack(note),
|
note: await this.noteEntityService.pack(note),
|
||||||
@@ -149,10 +149,6 @@ export class AntennaService implements OnApplicationShutdown {
|
|||||||
public async checkHitAntenna(antenna: Antenna, note: (Note | Packed<'Note'>), noteUser: { id: User['id']; username: string; host: string | null; }): Promise<boolean> {
|
public async checkHitAntenna(antenna: Antenna, note: (Note | Packed<'Note'>), noteUser: { id: User['id']; username: string; host: string | null; }): Promise<boolean> {
|
||||||
if (note.visibility === 'specified') return false;
|
if (note.visibility === 'specified') return false;
|
||||||
if (note.visibility === 'followers') return false;
|
if (note.visibility === 'followers') return false;
|
||||||
|
|
||||||
// アンテナ作成者がノート作成者にブロックされていたらスキップ
|
|
||||||
const blockings = await this.blockingCache.fetch(noteUser.id, () => this.blockingsRepository.findBy({ blockerId: noteUser.id }).then(res => res.map(x => x.blockeeId)));
|
|
||||||
if (blockings.some(blocking => blocking === antenna.userId)) return false;
|
|
||||||
|
|
||||||
if (!antenna.withReplies && note.replyId != null) return false;
|
if (!antenna.withReplies && note.replyId != null) return false;
|
||||||
|
|
||||||
|
@@ -21,18 +21,13 @@ export class CaptchaService {
|
|||||||
response,
|
response,
|
||||||
});
|
});
|
||||||
|
|
||||||
const res = await this.httpRequestService.fetch(
|
const res = await this.httpRequestService.send(url, {
|
||||||
url,
|
method: 'POST',
|
||||||
{
|
body: params.toString(),
|
||||||
method: 'POST',
|
headers: {
|
||||||
body: params,
|
'Content-Type': 'application/x-www-form-urlencoded',
|
||||||
},
|
},
|
||||||
{
|
}, { throwErrorWhenResponseNotOk: false });
|
||||||
noOkError: true,
|
|
||||||
}
|
|
||||||
).catch(err => {
|
|
||||||
throw `${err.message ?? err}`;
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!res.ok) {
|
if (!res.ok) {
|
||||||
throw `${res.status}`;
|
throw `${res.status}`;
|
||||||
|
@@ -62,7 +62,6 @@ import PerUserNotesChart from './chart/charts/per-user-notes.js';
|
|||||||
import PerUserPvChart from './chart/charts/per-user-pv.js';
|
import PerUserPvChart from './chart/charts/per-user-pv.js';
|
||||||
import DriveChart from './chart/charts/drive.js';
|
import DriveChart from './chart/charts/drive.js';
|
||||||
import PerUserReactionsChart from './chart/charts/per-user-reactions.js';
|
import PerUserReactionsChart from './chart/charts/per-user-reactions.js';
|
||||||
import HashtagChart from './chart/charts/hashtag.js';
|
|
||||||
import PerUserFollowingChart from './chart/charts/per-user-following.js';
|
import PerUserFollowingChart from './chart/charts/per-user-following.js';
|
||||||
import PerUserDriveChart from './chart/charts/per-user-drive.js';
|
import PerUserDriveChart from './chart/charts/per-user-drive.js';
|
||||||
import ApRequestChart from './chart/charts/ap-request.js';
|
import ApRequestChart from './chart/charts/ap-request.js';
|
||||||
@@ -187,7 +186,6 @@ const $PerUserNotesChart: Provider = { provide: 'PerUserNotesChart', useExisting
|
|||||||
const $PerUserPvChart: Provider = { provide: 'PerUserPvChart', useExisting: PerUserPvChart };
|
const $PerUserPvChart: Provider = { provide: 'PerUserPvChart', useExisting: PerUserPvChart };
|
||||||
const $DriveChart: Provider = { provide: 'DriveChart', useExisting: DriveChart };
|
const $DriveChart: Provider = { provide: 'DriveChart', useExisting: DriveChart };
|
||||||
const $PerUserReactionsChart: Provider = { provide: 'PerUserReactionsChart', useExisting: PerUserReactionsChart };
|
const $PerUserReactionsChart: Provider = { provide: 'PerUserReactionsChart', useExisting: PerUserReactionsChart };
|
||||||
const $HashtagChart: Provider = { provide: 'HashtagChart', useExisting: HashtagChart };
|
|
||||||
const $PerUserFollowingChart: Provider = { provide: 'PerUserFollowingChart', useExisting: PerUserFollowingChart };
|
const $PerUserFollowingChart: Provider = { provide: 'PerUserFollowingChart', useExisting: PerUserFollowingChart };
|
||||||
const $PerUserDriveChart: Provider = { provide: 'PerUserDriveChart', useExisting: PerUserDriveChart };
|
const $PerUserDriveChart: Provider = { provide: 'PerUserDriveChart', useExisting: PerUserDriveChart };
|
||||||
const $ApRequestChart: Provider = { provide: 'ApRequestChart', useExisting: ApRequestChart };
|
const $ApRequestChart: Provider = { provide: 'ApRequestChart', useExisting: ApRequestChart };
|
||||||
@@ -315,7 +313,6 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting
|
|||||||
PerUserPvChart,
|
PerUserPvChart,
|
||||||
DriveChart,
|
DriveChart,
|
||||||
PerUserReactionsChart,
|
PerUserReactionsChart,
|
||||||
HashtagChart,
|
|
||||||
PerUserFollowingChart,
|
PerUserFollowingChart,
|
||||||
PerUserDriveChart,
|
PerUserDriveChart,
|
||||||
ApRequestChart,
|
ApRequestChart,
|
||||||
@@ -437,7 +434,6 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting
|
|||||||
$PerUserPvChart,
|
$PerUserPvChart,
|
||||||
$DriveChart,
|
$DriveChart,
|
||||||
$PerUserReactionsChart,
|
$PerUserReactionsChart,
|
||||||
$HashtagChart,
|
|
||||||
$PerUserFollowingChart,
|
$PerUserFollowingChart,
|
||||||
$PerUserDriveChart,
|
$PerUserDriveChart,
|
||||||
$ApRequestChart,
|
$ApRequestChart,
|
||||||
@@ -559,7 +555,6 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting
|
|||||||
PerUserPvChart,
|
PerUserPvChart,
|
||||||
DriveChart,
|
DriveChart,
|
||||||
PerUserReactionsChart,
|
PerUserReactionsChart,
|
||||||
HashtagChart,
|
|
||||||
PerUserFollowingChart,
|
PerUserFollowingChart,
|
||||||
PerUserDriveChart,
|
PerUserDriveChart,
|
||||||
ApRequestChart,
|
ApRequestChart,
|
||||||
@@ -680,7 +675,6 @@ const $ApQuestionService: Provider = { provide: 'ApQuestionService', useExisting
|
|||||||
$PerUserPvChart,
|
$PerUserPvChart,
|
||||||
$DriveChart,
|
$DriveChart,
|
||||||
$PerUserReactionsChart,
|
$PerUserReactionsChart,
|
||||||
$HashtagChart,
|
|
||||||
$PerUserFollowingChart,
|
$PerUserFollowingChart,
|
||||||
$PerUserDriveChart,
|
$PerUserDriveChart,
|
||||||
$ApRequestChart,
|
$ApRequestChart,
|
||||||
|
@@ -26,7 +26,7 @@ export class CreateNotificationService {
|
|||||||
|
|
||||||
private notificationEntityService: NotificationEntityService,
|
private notificationEntityService: NotificationEntityService,
|
||||||
private idService: IdService,
|
private idService: IdService,
|
||||||
private globalEventServie: GlobalEventService,
|
private globalEventService: GlobalEventService,
|
||||||
private pushNotificationService: PushNotificationService,
|
private pushNotificationService: PushNotificationService,
|
||||||
) {
|
) {
|
||||||
}
|
}
|
||||||
@@ -60,7 +60,7 @@ export class CreateNotificationService {
|
|||||||
const packed = await this.notificationEntityService.pack(notification, {});
|
const packed = await this.notificationEntityService.pack(notification, {});
|
||||||
|
|
||||||
// Publish notification event
|
// Publish notification event
|
||||||
this.globalEventServie.publishMainStream(notifieeId, 'notification', packed);
|
this.globalEventService.publishMainStream(notifieeId, 'notification', packed);
|
||||||
|
|
||||||
// 2秒経っても(今回作成した)通知が既読にならなかったら「未読の通知がありますよ」イベントを発行する
|
// 2秒経っても(今回作成した)通知が既読にならなかったら「未読の通知がありますよ」イベントを発行する
|
||||||
setTimeout(async () => {
|
setTimeout(async () => {
|
||||||
@@ -77,7 +77,7 @@ export class CreateNotificationService {
|
|||||||
}
|
}
|
||||||
//#endregion
|
//#endregion
|
||||||
|
|
||||||
this.globalEventServie.publishMainStream(notifieeId, 'unreadNotification', packed);
|
this.globalEventService.publishMainStream(notifieeId, 'unreadNotification', packed);
|
||||||
this.pushNotificationService.pushNotification(notifieeId, 'notification', packed);
|
this.pushNotificationService.pushNotification(notifieeId, 'notification', packed);
|
||||||
|
|
||||||
if (type === 'follow') this.emailNotificationFollow(notifieeId, await this.usersRepository.findOneByOrFail({ id: data.notifierId! }));
|
if (type === 'follow') this.emailNotificationFollow(notifieeId, await this.usersRepository.findOneByOrFail({ id: data.notifierId! }));
|
||||||
|
@@ -2,22 +2,39 @@ import { Inject, Injectable } from '@nestjs/common';
|
|||||||
import { DataSource, In, IsNull } from 'typeorm';
|
import { DataSource, In, IsNull } from 'typeorm';
|
||||||
import { DI } from '@/di-symbols.js';
|
import { DI } from '@/di-symbols.js';
|
||||||
import { IdService } from '@/core/IdService.js';
|
import { IdService } from '@/core/IdService.js';
|
||||||
|
import { EmojiEntityService } from '@/core/entities/EmojiEntityService.js';
|
||||||
|
import { GlobalEventService } from '@/core/GlobalEventService.js';
|
||||||
import type { DriveFile } from '@/models/entities/DriveFile.js';
|
import type { DriveFile } from '@/models/entities/DriveFile.js';
|
||||||
import type { Emoji } from '@/models/entities/Emoji.js';
|
import type { Emoji } from '@/models/entities/Emoji.js';
|
||||||
import type { EmojisRepository } from '@/models/index.js';
|
import type { EmojisRepository, Note } from '@/models/index.js';
|
||||||
import { bindThis } from '@/decorators.js';
|
import { bindThis } from '@/decorators.js';
|
||||||
|
import { Cache } from '@/misc/cache.js';
|
||||||
|
import { UtilityService } from '@/core/UtilityService.js';
|
||||||
|
import type { Config } from '@/config.js';
|
||||||
|
import { ReactionService } from '@/core/ReactionService.js';
|
||||||
|
import { query } from '@/misc/prelude/url.js';
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class CustomEmojiService {
|
export class CustomEmojiService {
|
||||||
|
private cache: Cache<Emoji | null>;
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
|
@Inject(DI.config)
|
||||||
|
private config: Config,
|
||||||
|
|
||||||
@Inject(DI.db)
|
@Inject(DI.db)
|
||||||
private db: DataSource,
|
private db: DataSource,
|
||||||
|
|
||||||
@Inject(DI.emojisRepository)
|
@Inject(DI.emojisRepository)
|
||||||
private emojisRepository: EmojisRepository,
|
private emojisRepository: EmojisRepository,
|
||||||
|
|
||||||
|
private utilityService: UtilityService,
|
||||||
private idService: IdService,
|
private idService: IdService,
|
||||||
|
private emojiEntityService: EmojiEntityService,
|
||||||
|
private globalEventService: GlobalEventService,
|
||||||
|
private reactionService: ReactionService,
|
||||||
) {
|
) {
|
||||||
|
this.cache = new Cache<Emoji | null>(1000 * 60 * 60 * 12);
|
||||||
}
|
}
|
||||||
|
|
||||||
@bindThis
|
@bindThis
|
||||||
@@ -40,8 +57,135 @@ export class CustomEmojiService {
|
|||||||
type: data.driveFile.webpublicType ?? data.driveFile.type,
|
type: data.driveFile.webpublicType ?? data.driveFile.type,
|
||||||
}).then(x => this.emojisRepository.findOneByOrFail(x.identifiers[0]));
|
}).then(x => this.emojisRepository.findOneByOrFail(x.identifiers[0]));
|
||||||
|
|
||||||
await this.db.queryResultCache!.remove(['meta_emojis']);
|
if (data.host == null) {
|
||||||
|
await this.db.queryResultCache!.remove(['meta_emojis']);
|
||||||
|
|
||||||
|
this.globalEventService.publishBroadcastStream('emojiAdded', {
|
||||||
|
emoji: await this.emojiEntityService.pack(emoji.id),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
return emoji;
|
return emoji;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@bindThis
|
||||||
|
private normalizeHost(src: string | undefined, noteUserHost: string | null): string | null {
|
||||||
|
// クエリに使うホスト
|
||||||
|
let host = src === '.' ? null // .はローカルホスト (ここがマッチするのはリアクションのみ)
|
||||||
|
: src === undefined ? noteUserHost // ノートなどでホスト省略表記の場合はローカルホスト (ここがリアクションにマッチすることはない)
|
||||||
|
: this.utilityService.isSelfHost(src) ? null // 自ホスト指定
|
||||||
|
: (src || noteUserHost); // 指定されたホスト || ノートなどの所有者のホスト (こっちがリアクションにマッチすることはない)
|
||||||
|
|
||||||
|
host = this.utilityService.toPunyNullable(host);
|
||||||
|
|
||||||
|
return host;
|
||||||
|
}
|
||||||
|
|
||||||
|
@bindThis
|
||||||
|
private parseEmojiStr(emojiName: string, noteUserHost: string | null) {
|
||||||
|
const match = emojiName.match(/^(\w+)(?:@([\w.-]+))?$/);
|
||||||
|
if (!match) return { name: null, host: null };
|
||||||
|
|
||||||
|
const name = match[1];
|
||||||
|
|
||||||
|
// ホスト正規化
|
||||||
|
const host = this.utilityService.toPunyNullable(this.normalizeHost(match[2], noteUserHost));
|
||||||
|
|
||||||
|
return { name, host };
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 添付用(リモート)カスタム絵文字URLを解決する
|
||||||
|
* @param emojiName ノートやユーザープロフィールに添付された、またはリアクションのカスタム絵文字名 (:は含めない, リアクションでローカルホストの場合は@.を付ける (これはdecodeReactionで可能))
|
||||||
|
* @param noteUserHost ノートやユーザープロフィールの所有者のホスト
|
||||||
|
* @returns URL, nullは未マッチを意味する
|
||||||
|
*/
|
||||||
|
@bindThis
|
||||||
|
public async populateEmoji(emojiName: string, noteUserHost: string | null): Promise<string | null> {
|
||||||
|
const { name, host } = this.parseEmojiStr(emojiName, noteUserHost);
|
||||||
|
if (name == null) return null;
|
||||||
|
if (host == null) return null;
|
||||||
|
|
||||||
|
const queryOrNull = async () => (await this.emojisRepository.findOneBy({
|
||||||
|
name,
|
||||||
|
host: host ?? IsNull(),
|
||||||
|
})) ?? null;
|
||||||
|
|
||||||
|
const emoji = await this.cache.fetch(`${name} ${host}`, queryOrNull);
|
||||||
|
|
||||||
|
if (emoji == null) return null;
|
||||||
|
|
||||||
|
const isLocal = emoji.host == null;
|
||||||
|
const emojiUrl = emoji.publicUrl || emoji.originalUrl; // || emoji.originalUrl してるのは後方互換性のため(publicUrlはstringなので??はだめ)
|
||||||
|
const url = isLocal
|
||||||
|
? emojiUrl
|
||||||
|
: this.config.proxyRemoteFiles
|
||||||
|
? `${this.config.mediaProxy}/emoji.webp?${query({ url: emojiUrl })}`
|
||||||
|
: emojiUrl;
|
||||||
|
|
||||||
|
return url;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 複数の添付用(リモート)カスタム絵文字URLを解決する (キャシュ付き, 存在しないものは結果から除外される)
|
||||||
|
*/
|
||||||
|
@bindThis
|
||||||
|
public async populateEmojis(emojiNames: string[], noteUserHost: string | null): Promise<Record<string, string>> {
|
||||||
|
const emojis = await Promise.all(emojiNames.map(x => this.populateEmoji(x, noteUserHost)));
|
||||||
|
const res = {} as any;
|
||||||
|
for (let i = 0; i < emojiNames.length; i++) {
|
||||||
|
if (emojis[i] != null) {
|
||||||
|
res[emojiNames[i]] = emojis[i];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
@bindThis
|
||||||
|
public aggregateNoteEmojis(notes: Note[]) {
|
||||||
|
let emojis: { name: string | null; host: string | null; }[] = [];
|
||||||
|
for (const note of notes) {
|
||||||
|
emojis = emojis.concat(note.emojis
|
||||||
|
.map(e => this.parseEmojiStr(e, note.userHost)));
|
||||||
|
if (note.renote) {
|
||||||
|
emojis = emojis.concat(note.renote.emojis
|
||||||
|
.map(e => this.parseEmojiStr(e, note.renote!.userHost)));
|
||||||
|
if (note.renote.user) {
|
||||||
|
emojis = emojis.concat(note.renote.user.emojis
|
||||||
|
.map(e => this.parseEmojiStr(e, note.renote!.userHost)));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
const customReactions = Object.keys(note.reactions).map(x => this.reactionService.decodeReaction(x)).filter(x => x.name != null) as typeof emojis;
|
||||||
|
emojis = emojis.concat(customReactions);
|
||||||
|
if (note.user) {
|
||||||
|
emojis = emojis.concat(note.user.emojis
|
||||||
|
.map(e => this.parseEmojiStr(e, note.userHost)));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return emojis.filter(x => x.name != null && x.host != null) as { name: string; host: string; }[];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 与えられた絵文字のリストをデータベースから取得し、キャッシュに追加します
|
||||||
|
*/
|
||||||
|
@bindThis
|
||||||
|
public async prefetchEmojis(emojis: { name: string; host: string | null; }[]): Promise<void> {
|
||||||
|
const notCachedEmojis = emojis.filter(emoji => this.cache.get(`${emoji.name} ${emoji.host}`) == null);
|
||||||
|
const emojisQuery: any[] = [];
|
||||||
|
const hosts = new Set(notCachedEmojis.map(e => e.host));
|
||||||
|
for (const host of hosts) {
|
||||||
|
if (host == null) continue;
|
||||||
|
emojisQuery.push({
|
||||||
|
name: In(notCachedEmojis.filter(e => e.host === host).map(e => e.name)),
|
||||||
|
host: host,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
const _emojis = emojisQuery.length > 0 ? await this.emojisRepository.find({
|
||||||
|
where: emojisQuery,
|
||||||
|
select: ['name', 'host', 'originalUrl', 'publicUrl'],
|
||||||
|
}) : [];
|
||||||
|
for (const emoji of _emojis) {
|
||||||
|
this.cache.set(`${emoji.name} ${emoji.host}`, emoji);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@@ -14,7 +14,7 @@ export class DeleteAccountService {
|
|||||||
|
|
||||||
private userSuspendService: UserSuspendService,
|
private userSuspendService: UserSuspendService,
|
||||||
private queueService: QueueService,
|
private queueService: QueueService,
|
||||||
private globalEventServie: GlobalEventService,
|
private globalEventService: GlobalEventService,
|
||||||
) {
|
) {
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -38,6 +38,6 @@ export class DeleteAccountService {
|
|||||||
});
|
});
|
||||||
|
|
||||||
// Terminate streaming
|
// Terminate streaming
|
||||||
this.globalEventServie.publishUserEvent(user.id, 'terminate', {});
|
this.globalEventService.publishUserEvent(user.id, 'terminate', {});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -4,16 +4,15 @@ import * as util from 'node:util';
|
|||||||
import { Inject, Injectable } from '@nestjs/common';
|
import { Inject, Injectable } from '@nestjs/common';
|
||||||
import IPCIDR from 'ip-cidr';
|
import IPCIDR from 'ip-cidr';
|
||||||
import PrivateIp from 'private-ip';
|
import PrivateIp from 'private-ip';
|
||||||
import got, * as Got from 'got';
|
|
||||||
import chalk from 'chalk';
|
import chalk from 'chalk';
|
||||||
|
import got, * as Got from 'got';
|
||||||
import { DI } from '@/di-symbols.js';
|
import { DI } from '@/di-symbols.js';
|
||||||
import type { Config } from '@/config.js';
|
import type { Config } from '@/config.js';
|
||||||
import { HttpRequestService, UndiciFetcher } from '@/core/HttpRequestService.js';
|
import { HttpRequestService } from '@/core/HttpRequestService.js';
|
||||||
import { createTemp } from '@/misc/create-temp.js';
|
import { createTemp } from '@/misc/create-temp.js';
|
||||||
import { StatusError } from '@/misc/status-error.js';
|
import { StatusError } from '@/misc/status-error.js';
|
||||||
import { LoggerService } from '@/core/LoggerService.js';
|
import { LoggerService } from '@/core/LoggerService.js';
|
||||||
import type Logger from '@/logger.js';
|
import type Logger from '@/logger.js';
|
||||||
import { buildConnector } from 'undici';
|
|
||||||
|
|
||||||
const pipeline = util.promisify(stream.pipeline);
|
const pipeline = util.promisify(stream.pipeline);
|
||||||
import { bindThis } from '@/decorators.js';
|
import { bindThis } from '@/decorators.js';
|
||||||
@@ -21,7 +20,6 @@ import { bindThis } from '@/decorators.js';
|
|||||||
@Injectable()
|
@Injectable()
|
||||||
export class DownloadService {
|
export class DownloadService {
|
||||||
private logger: Logger;
|
private logger: Logger;
|
||||||
private undiciFetcher: UndiciFetcher;
|
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
@Inject(DI.config)
|
@Inject(DI.config)
|
||||||
@@ -31,24 +29,6 @@ export class DownloadService {
|
|||||||
private loggerService: LoggerService,
|
private loggerService: LoggerService,
|
||||||
) {
|
) {
|
||||||
this.logger = this.loggerService.getLogger('download');
|
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
|
@bindThis
|
||||||
@@ -59,14 +39,61 @@ export class DownloadService {
|
|||||||
const operationTimeout = 60 * 1000;
|
const operationTimeout = 60 * 1000;
|
||||||
const maxSize = this.config.maxFileSize ?? 262144000;
|
const maxSize = this.config.maxFileSize ?? 262144000;
|
||||||
|
|
||||||
const response = await this.undiciFetcher.fetch(url);
|
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,
|
||||||
|
},
|
||||||
|
enableUnixSockets: false,
|
||||||
|
}).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();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (response.body === null) {
|
const contentLength = res.headers['content-length'];
|
||||||
throw new StatusError('No body', 400, 'No body');
|
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;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
await pipeline(stream.Readable.fromWeb(response.body), fs.createWriteStream(path));
|
|
||||||
|
|
||||||
this.logger.succ(`Download finished: ${chalk.cyan(url)}`);
|
this.logger.succ(`Download finished: ${chalk.cyan(url)}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -2,6 +2,7 @@ import { URL } from 'node:url';
|
|||||||
import { Inject, Injectable } from '@nestjs/common';
|
import { Inject, Injectable } from '@nestjs/common';
|
||||||
import { JSDOM } from 'jsdom';
|
import { JSDOM } from 'jsdom';
|
||||||
import tinycolor from 'tinycolor2';
|
import tinycolor from 'tinycolor2';
|
||||||
|
import fetch from 'node-fetch';
|
||||||
import type { Instance } from '@/models/entities/Instance.js';
|
import type { Instance } from '@/models/entities/Instance.js';
|
||||||
import type { InstancesRepository } from '@/models/index.js';
|
import type { InstancesRepository } from '@/models/index.js';
|
||||||
import { AppLockService } from '@/core/AppLockService.js';
|
import { AppLockService } from '@/core/AppLockService.js';
|
||||||
@@ -190,7 +191,9 @@ export class FetchInstanceMetadataService {
|
|||||||
|
|
||||||
const faviconUrl = url + '/favicon.ico';
|
const faviconUrl = url + '/favicon.ico';
|
||||||
|
|
||||||
const favicon = await this.httpRequestService.fetch(faviconUrl, {}, { noOkError: true });
|
const favicon = await this.httpRequestService.send(faviconUrl, {
|
||||||
|
method: 'HEAD',
|
||||||
|
}, { throwErrorWhenResponseNotOk: false });
|
||||||
|
|
||||||
if (favicon.ok) {
|
if (favicon.ok) {
|
||||||
return faviconUrl;
|
return faviconUrl;
|
||||||
|
@@ -4,7 +4,6 @@ import type { User } from '@/models/entities/User.js';
|
|||||||
import { normalizeForSearch } from '@/misc/normalize-for-search.js';
|
import { normalizeForSearch } from '@/misc/normalize-for-search.js';
|
||||||
import { IdService } from '@/core/IdService.js';
|
import { IdService } from '@/core/IdService.js';
|
||||||
import type { Hashtag } from '@/models/entities/Hashtag.js';
|
import type { Hashtag } from '@/models/entities/Hashtag.js';
|
||||||
import HashtagChart from '@/core/chart/charts/hashtag.js';
|
|
||||||
import type { HashtagsRepository, UsersRepository } from '@/models/index.js';
|
import type { HashtagsRepository, UsersRepository } from '@/models/index.js';
|
||||||
import { UserEntityService } from '@/core/entities/UserEntityService.js';
|
import { UserEntityService } from '@/core/entities/UserEntityService.js';
|
||||||
import { bindThis } from '@/decorators.js';
|
import { bindThis } from '@/decorators.js';
|
||||||
@@ -20,7 +19,6 @@ export class HashtagService {
|
|||||||
|
|
||||||
private userEntityService: UserEntityService,
|
private userEntityService: UserEntityService,
|
||||||
private idService: IdService,
|
private idService: IdService,
|
||||||
private hashtagChart: HashtagChart,
|
|
||||||
) {
|
) {
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -143,9 +141,5 @@ export class HashtagService {
|
|||||||
} as Hashtag);
|
} as Hashtag);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!isUserAttached) {
|
|
||||||
this.hashtagChart.update(tag, user);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -1,257 +1,67 @@
|
|||||||
import * as http from 'node:http';
|
import * as http from 'node:http';
|
||||||
import * as https from 'node:https';
|
import * as https from 'node:https';
|
||||||
import CacheableLookup from 'cacheable-lookup';
|
import CacheableLookup from 'cacheable-lookup';
|
||||||
|
import fetch from 'node-fetch';
|
||||||
import { HttpProxyAgent, HttpsProxyAgent } from 'hpagent';
|
import { HttpProxyAgent, HttpsProxyAgent } from 'hpagent';
|
||||||
import { Inject, Injectable } from '@nestjs/common';
|
import { Inject, Injectable } from '@nestjs/common';
|
||||||
import { DI } from '@/di-symbols.js';
|
import { DI } from '@/di-symbols.js';
|
||||||
import type { Config } from '@/config.js';
|
import type { Config } from '@/config.js';
|
||||||
import { StatusError } from '@/misc/status-error.js';
|
import { StatusError } from '@/misc/status-error.js';
|
||||||
import { bindThis } from '@/decorators.js';
|
import { bindThis } from '@/decorators.js';
|
||||||
import * as undici from 'undici';
|
import type { Response } from 'node-fetch';
|
||||||
import { LookupFunction } from 'node:net';
|
import type { URL } from 'node:url';
|
||||||
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()
|
@Injectable()
|
||||||
export class HttpRequestService {
|
export class HttpRequestService {
|
||||||
public defaultFetcher: UndiciFetcher;
|
/**
|
||||||
public fetch: UndiciFetcher['fetch'];
|
* Get http non-proxy agent
|
||||||
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;
|
private http: http.Agent;
|
||||||
|
|
||||||
// https non-proxy agent
|
/**
|
||||||
|
* Get https non-proxy agent
|
||||||
|
*/
|
||||||
private https: https.Agent;
|
private https: https.Agent;
|
||||||
|
|
||||||
// http proxy or non-proxy agent
|
/**
|
||||||
|
* Get http proxy or non-proxy agent
|
||||||
|
*/
|
||||||
public httpAgent: http.Agent;
|
public httpAgent: http.Agent;
|
||||||
|
|
||||||
// https proxy or non-proxy agent
|
/**
|
||||||
|
* Get https proxy or non-proxy agent
|
||||||
|
*/
|
||||||
public httpsAgent: https.Agent;
|
public httpsAgent: https.Agent;
|
||||||
//#endregion
|
|
||||||
|
|
||||||
public readonly dnsCache: CacheableLookup;
|
|
||||||
public readonly clientDefaults: undici.Agent.Options;
|
|
||||||
private maxSockets: number;
|
|
||||||
|
|
||||||
private logger: Logger;
|
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
@Inject(DI.config)
|
@Inject(DI.config)
|
||||||
private config: Config,
|
private config: Config,
|
||||||
private loggerService: LoggerService,
|
|
||||||
) {
|
) {
|
||||||
this.logger = this.loggerService.getLogger('http-request');
|
const cache = new CacheableLookup({
|
||||||
|
|
||||||
this.dnsCache = new CacheableLookup({
|
|
||||||
maxTtl: 3600, // 1hours
|
maxTtl: 3600, // 1hours
|
||||||
errorTtl: 30, // 30secs
|
errorTtl: 30, // 30secs
|
||||||
lookup: false, // nativeのdns.lookupにfallbackしない
|
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({
|
this.http = new http.Agent({
|
||||||
keepAlive: true,
|
keepAlive: true,
|
||||||
keepAliveMsecs: 30 * 1000,
|
keepAliveMsecs: 30 * 1000,
|
||||||
lookup: this.dnsCache.lookup,
|
lookup: cache.lookup,
|
||||||
} as http.AgentOptions);
|
} as http.AgentOptions);
|
||||||
|
|
||||||
this.https = new https.Agent({
|
this.https = new https.Agent({
|
||||||
keepAlive: true,
|
keepAlive: true,
|
||||||
keepAliveMsecs: 30 * 1000,
|
keepAliveMsecs: 30 * 1000,
|
||||||
lookup: this.dnsCache.lookup,
|
lookup: cache.lookup,
|
||||||
} as https.AgentOptions);
|
} as https.AgentOptions);
|
||||||
|
|
||||||
|
const maxSockets = Math.max(256, config.deliverJobConcurrency ?? 128);
|
||||||
|
|
||||||
this.httpAgent = config.proxy
|
this.httpAgent = config.proxy
|
||||||
? new HttpProxyAgent({
|
? new HttpProxyAgent({
|
||||||
keepAlive: true,
|
keepAlive: true,
|
||||||
keepAliveMsecs: 30 * 1000,
|
keepAliveMsecs: 30 * 1000,
|
||||||
maxSockets: this.maxSockets,
|
maxSockets,
|
||||||
maxFreeSockets: 256,
|
maxFreeSockets: 256,
|
||||||
scheduling: 'lifo',
|
scheduling: 'lifo',
|
||||||
proxy: config.proxy,
|
proxy: config.proxy,
|
||||||
@@ -262,42 +72,21 @@ export class HttpRequestService {
|
|||||||
? new HttpsProxyAgent({
|
? new HttpsProxyAgent({
|
||||||
keepAlive: true,
|
keepAlive: true,
|
||||||
keepAliveMsecs: 30 * 1000,
|
keepAliveMsecs: 30 * 1000,
|
||||||
maxSockets: this.maxSockets,
|
maxSockets,
|
||||||
maxFreeSockets: 256,
|
maxFreeSockets: 256,
|
||||||
scheduling: 'lifo',
|
scheduling: 'lifo',
|
||||||
proxy: config.proxy,
|
proxy: config.proxy,
|
||||||
})
|
})
|
||||||
: this.https;
|
: 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 http agent by URL
|
* Get agent by URL
|
||||||
* @param url URL
|
* @param url URL
|
||||||
* @param bypassProxy Allways bypass proxy
|
* @param bypassProxy Allways bypass proxy
|
||||||
*/
|
*/
|
||||||
@bindThis
|
@bindThis
|
||||||
public getHttpAgentByUrl(url: URL, bypassProxy = false): http.Agent | https.Agent {
|
public getAgentByUrl(url: URL, bypassProxy = false): http.Agent | https.Agent {
|
||||||
if (bypassProxy || (this.config.proxyBypassHosts || []).includes(url.hostname)) {
|
if (bypassProxy || (this.config.proxyBypassHosts || []).includes(url.hostname)) {
|
||||||
return url.protocol === 'http:' ? this.http : this.https;
|
return url.protocol === 'http:' ? this.http : this.https;
|
||||||
} else {
|
} else {
|
||||||
@@ -305,37 +94,67 @@ export class HttpRequestService {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* check ip
|
|
||||||
*/
|
|
||||||
@bindThis
|
@bindThis
|
||||||
public getConnectorWithIpCheck(connector: undici.buildConnector.connector, checkIp: IpChecker): undici.buildConnector.connectorAsync {
|
public async getJson<T = unknown>(url: string, accept = 'application/json, */*', headers?: Record<string, string>): Promise<T> {
|
||||||
return (options, cb) => {
|
const res = await this.send(url, {
|
||||||
connector(options, (err, socket) => {
|
method: 'GET',
|
||||||
this.logger.debug('Socket connector (with ip checker) called', socket);
|
headers: Object.assign({
|
||||||
if (err) {
|
'User-Agent': this.config.userAgent,
|
||||||
this.logger.error(`Socket error`, err)
|
Accept: accept,
|
||||||
cb(new Error(`Error while socket connecting\n${err}`), null);
|
}, headers ?? {}),
|
||||||
return;
|
timeout: 5000,
|
||||||
}
|
size: 1024 * 256,
|
||||||
|
});
|
||||||
|
|
||||||
if (socket.remoteAddress == undefined) {
|
return await res.json() as T;
|
||||||
this.logger.error(`Socket error: remoteAddress is undefined`);
|
}
|
||||||
cb(new Error('remoteAddress is undefined (maybe socket destroyed)'), null);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// allow
|
@bindThis
|
||||||
if (checkIp(socket.remoteAddress)) {
|
public async getHtml(url: string, accept = 'text/html, */*', headers?: Record<string, string>): Promise<string> {
|
||||||
this.logger.debug(`Socket connected (ip ok): ${socket.localPort} => ${socket.remoteAddress}`);
|
const res = await this.send(url, {
|
||||||
cb(null, socket);
|
method: 'GET',
|
||||||
return;
|
headers: Object.assign({
|
||||||
}
|
'User-Agent': this.config.userAgent,
|
||||||
|
Accept: accept,
|
||||||
|
}, headers ?? {}),
|
||||||
|
timeout: 5000,
|
||||||
|
});
|
||||||
|
|
||||||
this.logger.error('IP is not allowed', socket);
|
return await res.text();
|
||||||
cb(new StatusError('IP is not allowed', 403, 'IP is not allowed'), null);
|
}
|
||||||
socket.destroy();
|
|
||||||
});
|
@bindThis
|
||||||
};
|
public async send(url: string, args: {
|
||||||
|
method?: string,
|
||||||
|
body?: string,
|
||||||
|
headers?: Record<string, string>,
|
||||||
|
timeout?: number,
|
||||||
|
size?: number,
|
||||||
|
} = {}, extra: {
|
||||||
|
throwErrorWhenResponseNotOk: boolean;
|
||||||
|
} = {
|
||||||
|
throwErrorWhenResponseNotOk: true,
|
||||||
|
}): Promise<Response> {
|
||||||
|
const timeout = args.timeout ?? 5000;
|
||||||
|
|
||||||
|
const controller = new AbortController();
|
||||||
|
setTimeout(() => {
|
||||||
|
controller.abort();
|
||||||
|
}, timeout);
|
||||||
|
|
||||||
|
const res = await fetch(url, {
|
||||||
|
method: args.method ?? 'GET',
|
||||||
|
headers: args.headers,
|
||||||
|
body: args.body,
|
||||||
|
size: args.size ?? 10 * 1024 * 1024,
|
||||||
|
agent: (url) => this.getAgentByUrl(url),
|
||||||
|
signal: controller.signal,
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!res.ok && extra.throwErrorWhenResponseNotOk) {
|
||||||
|
throw new StatusError(`${res.status} ${res.statusText}`, res.status, res.statusText);
|
||||||
|
}
|
||||||
|
|
||||||
|
return res;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -9,6 +9,14 @@ export type IImage = {
|
|||||||
type: string;
|
type: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export type IImageStream = {
|
||||||
|
data: Readable;
|
||||||
|
ext: string | null;
|
||||||
|
type: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type IImageStreamable = IImage | IImageStream;
|
||||||
|
|
||||||
export const webpDefault: sharp.WebpOptions = {
|
export const webpDefault: sharp.WebpOptions = {
|
||||||
quality: 85,
|
quality: 85,
|
||||||
alphaQuality: 95,
|
alphaQuality: 95,
|
||||||
@@ -19,6 +27,7 @@ export const webpDefault: sharp.WebpOptions = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
import { bindThis } from '@/decorators.js';
|
import { bindThis } from '@/decorators.js';
|
||||||
|
import { Readable } from 'node:stream';
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class ImageProcessingService {
|
export class ImageProcessingService {
|
||||||
@@ -64,7 +73,7 @@ export class ImageProcessingService {
|
|||||||
*/
|
*/
|
||||||
@bindThis
|
@bindThis
|
||||||
public async convertToWebp(path: string, width: number, height: number, options: sharp.WebpOptions = webpDefault): Promise<IImage> {
|
public async convertToWebp(path: string, width: number, height: number, options: sharp.WebpOptions = webpDefault): Promise<IImage> {
|
||||||
return this.convertSharpToWebp(await sharp(path), width, height, options);
|
return this.convertSharpToWebp(sharp(path), width, height, options);
|
||||||
}
|
}
|
||||||
|
|
||||||
@bindThis
|
@bindThis
|
||||||
@@ -85,6 +94,27 @@ export class ImageProcessingService {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@bindThis
|
||||||
|
public convertToWebpStream(path: string, width: number, height: number, options: sharp.WebpOptions = webpDefault): IImageStream {
|
||||||
|
return this.convertSharpToWebpStream(sharp(path), width, height, options);
|
||||||
|
}
|
||||||
|
|
||||||
|
@bindThis
|
||||||
|
public convertSharpToWebpStream(sharp: sharp.Sharp, width: number, height: number, options: sharp.WebpOptions = webpDefault): IImageStream {
|
||||||
|
const data = sharp
|
||||||
|
.resize(width, height, {
|
||||||
|
fit: 'inside',
|
||||||
|
withoutEnlargement: true,
|
||||||
|
})
|
||||||
|
.rotate()
|
||||||
|
.webp(options)
|
||||||
|
|
||||||
|
return {
|
||||||
|
data,
|
||||||
|
ext: 'webp',
|
||||||
|
type: 'image/webp',
|
||||||
|
};
|
||||||
|
}
|
||||||
/**
|
/**
|
||||||
* Convert to PNG
|
* Convert to PNG
|
||||||
* with resize, remove metadata, resolve orientation, stop animation
|
* with resize, remove metadata, resolve orientation, stop animation
|
||||||
|
@@ -1,5 +1,4 @@
|
|||||||
import { Inject, Injectable } from '@nestjs/common';
|
import { Inject, Injectable } from '@nestjs/common';
|
||||||
import * as SyslogPro from 'syslog-pro';
|
|
||||||
import { DI } from '@/di-symbols.js';
|
import { DI } from '@/di-symbols.js';
|
||||||
import type { Config } from '@/config.js';
|
import type { Config } from '@/config.js';
|
||||||
import Logger from '@/logger.js';
|
import Logger from '@/logger.js';
|
||||||
@@ -8,29 +7,14 @@ import type { KEYWORD } from 'color-convert/conversions';
|
|||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class LoggerService {
|
export class LoggerService {
|
||||||
private syslogClient;
|
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
@Inject(DI.config)
|
@Inject(DI.config)
|
||||||
private config: Config,
|
private config: Config,
|
||||||
) {
|
) {
|
||||||
if (this.config.syslog) {
|
|
||||||
this.syslogClient = new SyslogPro.RFC5424({
|
|
||||||
applicationName: 'Misskey',
|
|
||||||
timestamp: true,
|
|
||||||
includeStructuredData: true,
|
|
||||||
color: true,
|
|
||||||
extendedColor: true,
|
|
||||||
server: {
|
|
||||||
target: config.syslog.host,
|
|
||||||
port: config.syslog.port,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@bindThis
|
@bindThis
|
||||||
public getLogger(domain: string, color?: KEYWORD | undefined, store?: boolean) {
|
public getLogger(domain: string, color?: KEYWORD | undefined, store?: boolean) {
|
||||||
return new Logger(domain, color, store, this.syslogClient);
|
return new Logger(domain, color, store);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -175,7 +175,7 @@ export class NoteCreateService {
|
|||||||
private userEntityService: UserEntityService,
|
private userEntityService: UserEntityService,
|
||||||
private noteEntityService: NoteEntityService,
|
private noteEntityService: NoteEntityService,
|
||||||
private idService: IdService,
|
private idService: IdService,
|
||||||
private globalEventServie: GlobalEventService,
|
private globalEventService: GlobalEventService,
|
||||||
private queueService: QueueService,
|
private queueService: QueueService,
|
||||||
private noteReadService: NoteReadService,
|
private noteReadService: NoteReadService,
|
||||||
private createNotificationService: CreateNotificationService,
|
private createNotificationService: CreateNotificationService,
|
||||||
@@ -535,7 +535,7 @@ export class NoteCreateService {
|
|||||||
// Pack the note
|
// Pack the note
|
||||||
const noteObj = await this.noteEntityService.pack(note);
|
const noteObj = await this.noteEntityService.pack(note);
|
||||||
|
|
||||||
this.globalEventServie.publishNotesStream(noteObj);
|
this.globalEventService.publishNotesStream(noteObj);
|
||||||
|
|
||||||
this.webhookService.getActiveWebhooks().then(webhooks => {
|
this.webhookService.getActiveWebhooks().then(webhooks => {
|
||||||
webhooks = webhooks.filter(x => x.userId === user.id && x.on.includes('note'));
|
webhooks = webhooks.filter(x => x.userId === user.id && x.on.includes('note'));
|
||||||
@@ -561,7 +561,7 @@ export class NoteCreateService {
|
|||||||
|
|
||||||
if (!threadMuted) {
|
if (!threadMuted) {
|
||||||
nm.push(data.reply.userId, 'reply');
|
nm.push(data.reply.userId, 'reply');
|
||||||
this.globalEventServie.publishMainStream(data.reply.userId, 'reply', noteObj);
|
this.globalEventService.publishMainStream(data.reply.userId, 'reply', noteObj);
|
||||||
|
|
||||||
const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === data.reply!.userId && x.on.includes('reply'));
|
const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === data.reply!.userId && x.on.includes('reply'));
|
||||||
for (const webhook of webhooks) {
|
for (const webhook of webhooks) {
|
||||||
@@ -584,7 +584,7 @@ export class NoteCreateService {
|
|||||||
|
|
||||||
// Publish event
|
// Publish event
|
||||||
if ((user.id !== data.renote.userId) && data.renote.userHost === null) {
|
if ((user.id !== data.renote.userId) && data.renote.userHost === null) {
|
||||||
this.globalEventServie.publishMainStream(data.renote.userId, 'renote', noteObj);
|
this.globalEventService.publishMainStream(data.renote.userId, 'renote', noteObj);
|
||||||
|
|
||||||
const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === data.renote!.userId && x.on.includes('renote'));
|
const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === data.renote!.userId && x.on.includes('renote'));
|
||||||
for (const webhook of webhooks) {
|
for (const webhook of webhooks) {
|
||||||
@@ -684,7 +684,7 @@ export class NoteCreateService {
|
|||||||
detail: true,
|
detail: true,
|
||||||
});
|
});
|
||||||
|
|
||||||
this.globalEventServie.publishMainStream(u.id, 'mention', detailPackedNote);
|
this.globalEventService.publishMainStream(u.id, 'mention', detailPackedNote);
|
||||||
|
|
||||||
const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === u.id && x.on.includes('mention'));
|
const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === u.id && x.on.includes('mention'));
|
||||||
for (const webhook of webhooks) {
|
for (const webhook of webhooks) {
|
||||||
|
@@ -34,7 +34,7 @@ export class NoteDeleteService {
|
|||||||
|
|
||||||
private userEntityService: UserEntityService,
|
private userEntityService: UserEntityService,
|
||||||
private noteEntityService: NoteEntityService,
|
private noteEntityService: NoteEntityService,
|
||||||
private globalEventServie: GlobalEventService,
|
private globalEventService: GlobalEventService,
|
||||||
private relayService: RelayService,
|
private relayService: RelayService,
|
||||||
private federatedInstanceService: FederatedInstanceService,
|
private federatedInstanceService: FederatedInstanceService,
|
||||||
private apRendererService: ApRendererService,
|
private apRendererService: ApRendererService,
|
||||||
@@ -63,7 +63,7 @@ export class NoteDeleteService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (!quiet) {
|
if (!quiet) {
|
||||||
this.globalEventServie.publishNoteStream(note.id, 'deleted', {
|
this.globalEventService.publishNoteStream(note.id, 'deleted', {
|
||||||
deletedAt: deletedAt,
|
deletedAt: deletedAt,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@@ -9,9 +9,9 @@ import { IdService } from '@/core/IdService.js';
|
|||||||
import { GlobalEventService } from '@/core/GlobalEventService.js';
|
import { GlobalEventService } from '@/core/GlobalEventService.js';
|
||||||
import type { UsersRepository, NoteUnreadsRepository, MutingsRepository, NoteThreadMutingsRepository, FollowingsRepository, ChannelFollowingsRepository, AntennaNotesRepository } from '@/models/index.js';
|
import type { UsersRepository, NoteUnreadsRepository, MutingsRepository, NoteThreadMutingsRepository, FollowingsRepository, ChannelFollowingsRepository, AntennaNotesRepository } from '@/models/index.js';
|
||||||
import { UserEntityService } from '@/core/entities/UserEntityService.js';
|
import { UserEntityService } from '@/core/entities/UserEntityService.js';
|
||||||
|
import { bindThis } from '@/decorators.js';
|
||||||
import { NotificationService } from './NotificationService.js';
|
import { NotificationService } from './NotificationService.js';
|
||||||
import { AntennaService } from './AntennaService.js';
|
import { AntennaService } from './AntennaService.js';
|
||||||
import { bindThis } from '@/decorators.js';
|
|
||||||
import { PushNotificationService } from './PushNotificationService.js';
|
import { PushNotificationService } from './PushNotificationService.js';
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
@@ -40,7 +40,7 @@ export class NoteReadService {
|
|||||||
|
|
||||||
private userEntityService: UserEntityService,
|
private userEntityService: UserEntityService,
|
||||||
private idService: IdService,
|
private idService: IdService,
|
||||||
private globalEventServie: GlobalEventService,
|
private globalEventService: GlobalEventService,
|
||||||
private notificationService: NotificationService,
|
private notificationService: NotificationService,
|
||||||
private antennaService: AntennaService,
|
private antennaService: AntennaService,
|
||||||
private pushNotificationService: PushNotificationService,
|
private pushNotificationService: PushNotificationService,
|
||||||
@@ -87,13 +87,13 @@ export class NoteReadService {
|
|||||||
if (exist == null) return;
|
if (exist == null) return;
|
||||||
|
|
||||||
if (params.isMentioned) {
|
if (params.isMentioned) {
|
||||||
this.globalEventServie.publishMainStream(userId, 'unreadMention', note.id);
|
this.globalEventService.publishMainStream(userId, 'unreadMention', note.id);
|
||||||
}
|
}
|
||||||
if (params.isSpecified) {
|
if (params.isSpecified) {
|
||||||
this.globalEventServie.publishMainStream(userId, 'unreadSpecifiedNote', note.id);
|
this.globalEventService.publishMainStream(userId, 'unreadSpecifiedNote', note.id);
|
||||||
}
|
}
|
||||||
if (note.channelId) {
|
if (note.channelId) {
|
||||||
this.globalEventServie.publishMainStream(userId, 'unreadChannel', note.id);
|
this.globalEventService.publishMainStream(userId, 'unreadChannel', note.id);
|
||||||
}
|
}
|
||||||
}, 2000);
|
}, 2000);
|
||||||
}
|
}
|
||||||
@@ -107,12 +107,6 @@ export class NoteReadService {
|
|||||||
followingChannels: Set<Channel['id']>;
|
followingChannels: Set<Channel['id']>;
|
||||||
},
|
},
|
||||||
): Promise<void> {
|
): Promise<void> {
|
||||||
const following = info?.following ? info.following : new Set<string>((await this.followingsRepository.find({
|
|
||||||
where: {
|
|
||||||
followerId: userId,
|
|
||||||
},
|
|
||||||
select: ['followeeId'],
|
|
||||||
})).map(x => x.followeeId));
|
|
||||||
const followingChannels = info?.followingChannels ? info.followingChannels : new Set<string>((await this.channelFollowingsRepository.find({
|
const followingChannels = info?.followingChannels ? info.followingChannels : new Set<string>((await this.channelFollowingsRepository.find({
|
||||||
where: {
|
where: {
|
||||||
followerId: userId,
|
followerId: userId,
|
||||||
@@ -139,7 +133,7 @@ export class NoteReadService {
|
|||||||
|
|
||||||
if (note.user != null) { // たぶんnullになることは無いはずだけど一応
|
if (note.user != null) { // たぶんnullになることは無いはずだけど一応
|
||||||
for (const antenna of myAntennas) {
|
for (const antenna of myAntennas) {
|
||||||
if (await this.antennaService.checkHitAntenna(antenna, note, note.user, undefined, Array.from(following))) {
|
if (await this.antennaService.checkHitAntenna(antenna, note, note.user)) {
|
||||||
readAntennaNotes.push(note);
|
readAntennaNotes.push(note);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -161,7 +155,7 @@ export class NoteReadService {
|
|||||||
}).then(mentionsCount => {
|
}).then(mentionsCount => {
|
||||||
if (mentionsCount === 0) {
|
if (mentionsCount === 0) {
|
||||||
// 全て既読になったイベントを発行
|
// 全て既読になったイベントを発行
|
||||||
this.globalEventServie.publishMainStream(userId, 'readAllUnreadMentions');
|
this.globalEventService.publishMainStream(userId, 'readAllUnreadMentions');
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -171,7 +165,7 @@ export class NoteReadService {
|
|||||||
}).then(specifiedCount => {
|
}).then(specifiedCount => {
|
||||||
if (specifiedCount === 0) {
|
if (specifiedCount === 0) {
|
||||||
// 全て既読になったイベントを発行
|
// 全て既読になったイベントを発行
|
||||||
this.globalEventServie.publishMainStream(userId, 'readAllUnreadSpecifiedNotes');
|
this.globalEventService.publishMainStream(userId, 'readAllUnreadSpecifiedNotes');
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -181,7 +175,7 @@ export class NoteReadService {
|
|||||||
}).then(channelNoteCount => {
|
}).then(channelNoteCount => {
|
||||||
if (channelNoteCount === 0) {
|
if (channelNoteCount === 0) {
|
||||||
// 全て既読になったイベントを発行
|
// 全て既読になったイベントを発行
|
||||||
this.globalEventServie.publishMainStream(userId, 'readAllChannels');
|
this.globalEventService.publishMainStream(userId, 'readAllChannels');
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -206,14 +200,14 @@ export class NoteReadService {
|
|||||||
});
|
});
|
||||||
|
|
||||||
if (count === 0) {
|
if (count === 0) {
|
||||||
this.globalEventServie.publishMainStream(userId, 'readAntenna', antenna);
|
this.globalEventService.publishMainStream(userId, 'readAntenna', antenna);
|
||||||
this.pushNotificationService.pushNotification(userId, 'readAntenna', { antennaId: antenna.id });
|
this.pushNotificationService.pushNotification(userId, 'readAntenna', { antennaId: antenna.id });
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
this.userEntityService.getHasUnreadAntenna(userId).then(unread => {
|
this.userEntityService.getHasUnreadAntenna(userId).then(unread => {
|
||||||
if (!unread) {
|
if (!unread) {
|
||||||
this.globalEventServie.publishMainStream(userId, 'readAllAntennas');
|
this.globalEventService.publishMainStream(userId, 'readAllAntennas');
|
||||||
this.pushNotificationService.pushNotification(userId, 'readAllAntennas', undefined);
|
this.pushNotificationService.pushNotification(userId, 'readAllAntennas', undefined);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@@ -1,17 +1,17 @@
|
|||||||
import { Inject, Injectable } from '@nestjs/common';
|
import { Inject, Injectable } from '@nestjs/common';
|
||||||
import { Not } from 'typeorm';
|
import { Not } from 'typeorm';
|
||||||
import { DI } from '@/di-symbols.js';
|
import { DI } from '@/di-symbols.js';
|
||||||
import type { NotesRepository, UsersRepository, BlockingsRepository, PollsRepository, PollVotesRepository } from '@/models/index.js';
|
import type { NotesRepository, UsersRepository, PollsRepository, PollVotesRepository } from '@/models/index.js';
|
||||||
import type { Note } from '@/models/entities/Note.js';
|
import type { Note } from '@/models/entities/Note.js';
|
||||||
import { RelayService } from '@/core/RelayService.js';
|
import { RelayService } from '@/core/RelayService.js';
|
||||||
import type { CacheableUser } from '@/models/entities/User.js';
|
import type { CacheableUser } from '@/models/entities/User.js';
|
||||||
import { IdService } from '@/core/IdService.js';
|
import { IdService } from '@/core/IdService.js';
|
||||||
import { GlobalEventService } from '@/core/GlobalEventService.js';
|
import { GlobalEventService } from '@/core/GlobalEventService.js';
|
||||||
import { CreateNotificationService } from '@/core/CreateNotificationService.js';
|
|
||||||
import { ApRendererService } from '@/core/activitypub/ApRendererService.js';
|
import { ApRendererService } from '@/core/activitypub/ApRendererService.js';
|
||||||
import { UserEntityService } from '@/core/entities/UserEntityService.js';
|
import { UserEntityService } from '@/core/entities/UserEntityService.js';
|
||||||
import { ApDeliverManagerService } from '@/core/activitypub/ApDeliverManagerService.js';
|
import { ApDeliverManagerService } from '@/core/activitypub/ApDeliverManagerService.js';
|
||||||
import { bindThis } from '@/decorators.js';
|
import { bindThis } from '@/decorators.js';
|
||||||
|
import { UserBlockingService } from '@/core/UserBlockingService.js';
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class PollService {
|
export class PollService {
|
||||||
@@ -28,14 +28,11 @@ export class PollService {
|
|||||||
@Inject(DI.pollVotesRepository)
|
@Inject(DI.pollVotesRepository)
|
||||||
private pollVotesRepository: PollVotesRepository,
|
private pollVotesRepository: PollVotesRepository,
|
||||||
|
|
||||||
@Inject(DI.blockingsRepository)
|
|
||||||
private blockingsRepository: BlockingsRepository,
|
|
||||||
|
|
||||||
private userEntityService: UserEntityService,
|
private userEntityService: UserEntityService,
|
||||||
private idService: IdService,
|
private idService: IdService,
|
||||||
private relayService: RelayService,
|
private relayService: RelayService,
|
||||||
private globalEventServie: GlobalEventService,
|
private globalEventService: GlobalEventService,
|
||||||
private createNotificationService: CreateNotificationService,
|
private userBlockingService: UserBlockingService,
|
||||||
private apRendererService: ApRendererService,
|
private apRendererService: ApRendererService,
|
||||||
private apDeliverManagerService: ApDeliverManagerService,
|
private apDeliverManagerService: ApDeliverManagerService,
|
||||||
) {
|
) {
|
||||||
@@ -52,11 +49,8 @@ export class PollService {
|
|||||||
|
|
||||||
// Check blocking
|
// Check blocking
|
||||||
if (note.userId !== user.id) {
|
if (note.userId !== user.id) {
|
||||||
const block = await this.blockingsRepository.findOneBy({
|
const blocked = await this.userBlockingService.checkBlocked(note.userId, user.id);
|
||||||
blockerId: note.userId,
|
if (blocked) {
|
||||||
blockeeId: user.id,
|
|
||||||
});
|
|
||||||
if (block) {
|
|
||||||
throw new Error('blocked');
|
throw new Error('blocked');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -88,7 +82,7 @@ export class PollService {
|
|||||||
const index = choice + 1; // In SQL, array index is 1 based
|
const index = choice + 1; // In SQL, array index is 1 based
|
||||||
await this.pollsRepository.query(`UPDATE poll SET votes[${index}] = votes[${index}] + 1 WHERE "noteId" = '${poll.noteId}'`);
|
await this.pollsRepository.query(`UPDATE poll SET votes[${index}] = votes[${index}] + 1 WHERE "noteId" = '${poll.noteId}'`);
|
||||||
|
|
||||||
this.globalEventServie.publishNoteStream(note.id, 'pollVoted', {
|
this.globalEventService.publishNoteStream(note.id, 'pollVoted', {
|
||||||
choice: choice,
|
choice: choice,
|
||||||
userId: user.id,
|
userId: user.id,
|
||||||
});
|
});
|
||||||
|
@@ -1,10 +1,10 @@
|
|||||||
import { Inject, Injectable } from '@nestjs/common';
|
import { Inject, Injectable } from '@nestjs/common';
|
||||||
import { Brackets } from 'typeorm';
|
import { Brackets, ObjectLiteral } from 'typeorm';
|
||||||
import { DI } from '@/di-symbols.js';
|
import { DI } from '@/di-symbols.js';
|
||||||
import type { User } from '@/models/entities/User.js';
|
import type { User } from '@/models/entities/User.js';
|
||||||
import type { UserProfilesRepository, FollowingsRepository, ChannelFollowingsRepository, MutedNotesRepository, BlockingsRepository, NoteThreadMutingsRepository, MutingsRepository } from '@/models/index.js';
|
import type { UserProfilesRepository, FollowingsRepository, ChannelFollowingsRepository, MutedNotesRepository, BlockingsRepository, NoteThreadMutingsRepository, MutingsRepository } from '@/models/index.js';
|
||||||
import type { SelectQueryBuilder } from 'typeorm';
|
|
||||||
import { bindThis } from '@/decorators.js';
|
import { bindThis } from '@/decorators.js';
|
||||||
|
import type { SelectQueryBuilder } from 'typeorm';
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class QueryService {
|
export class QueryService {
|
||||||
@@ -32,7 +32,7 @@ export class QueryService {
|
|||||||
) {
|
) {
|
||||||
}
|
}
|
||||||
|
|
||||||
public makePaginationQuery<T>(q: SelectQueryBuilder<T>, sinceId?: string, untilId?: string, sinceDate?: number, untilDate?: number): SelectQueryBuilder<T> {
|
public makePaginationQuery<T extends ObjectLiteral>(q: SelectQueryBuilder<T>, sinceId?: string, untilId?: string, sinceDate?: number, untilDate?: number): SelectQueryBuilder<T> {
|
||||||
if (sinceId && untilId) {
|
if (sinceId && untilId) {
|
||||||
q.andWhere(`${q.alias}.id > :sinceId`, { sinceId: sinceId });
|
q.andWhere(`${q.alias}.id > :sinceId`, { sinceId: sinceId });
|
||||||
q.andWhere(`${q.alias}.id < :untilId`, { untilId: untilId });
|
q.andWhere(`${q.alias}.id < :untilId`, { untilId: untilId });
|
||||||
|
@@ -18,7 +18,8 @@ import { UserEntityService } from '@/core/entities/UserEntityService.js';
|
|||||||
import { ApRendererService } from '@/core/activitypub/ApRendererService.js';
|
import { ApRendererService } from '@/core/activitypub/ApRendererService.js';
|
||||||
import { MetaService } from '@/core/MetaService.js';
|
import { MetaService } from '@/core/MetaService.js';
|
||||||
import { bindThis } from '@/decorators.js';
|
import { bindThis } from '@/decorators.js';
|
||||||
import { UtilityService } from './UtilityService.js';
|
import { UtilityService } from '@/core/UtilityService.js';
|
||||||
|
import { UserBlockingService } from '@/core/UserBlockingService.js';
|
||||||
|
|
||||||
const legacies: Record<string, string> = {
|
const legacies: Record<string, string> = {
|
||||||
'like': '👍',
|
'like': '👍',
|
||||||
@@ -73,8 +74,9 @@ export class ReactionService {
|
|||||||
private metaService: MetaService,
|
private metaService: MetaService,
|
||||||
private userEntityService: UserEntityService,
|
private userEntityService: UserEntityService,
|
||||||
private noteEntityService: NoteEntityService,
|
private noteEntityService: NoteEntityService,
|
||||||
|
private userBlockingService: UserBlockingService,
|
||||||
private idService: IdService,
|
private idService: IdService,
|
||||||
private globalEventServie: GlobalEventService,
|
private globalEventService: GlobalEventService,
|
||||||
private apRendererService: ApRendererService,
|
private apRendererService: ApRendererService,
|
||||||
private apDeliverManagerService: ApDeliverManagerService,
|
private apDeliverManagerService: ApDeliverManagerService,
|
||||||
private createNotificationService: CreateNotificationService,
|
private createNotificationService: CreateNotificationService,
|
||||||
@@ -86,11 +88,8 @@ export class ReactionService {
|
|||||||
public async create(user: { id: User['id']; host: User['host']; isBot: User['isBot'] }, note: Note, reaction?: string) {
|
public async create(user: { id: User['id']; host: User['host']; isBot: User['isBot'] }, note: Note, reaction?: string) {
|
||||||
// Check blocking
|
// Check blocking
|
||||||
if (note.userId !== user.id) {
|
if (note.userId !== user.id) {
|
||||||
const block = await this.blockingsRepository.findOneBy({
|
const blocked = await this.userBlockingService.checkBlocked(note.userId, user.id);
|
||||||
blockerId: note.userId,
|
if (blocked) {
|
||||||
blockeeId: user.id,
|
|
||||||
});
|
|
||||||
if (block) {
|
|
||||||
throw new IdentifiableError('e70412a4-7197-4726-8e74-f3e0deb92aa7');
|
throw new IdentifiableError('e70412a4-7197-4726-8e74-f3e0deb92aa7');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -157,7 +156,7 @@ export class ReactionService {
|
|||||||
select: ['name', 'host', 'originalUrl', 'publicUrl'],
|
select: ['name', 'host', 'originalUrl', 'publicUrl'],
|
||||||
});
|
});
|
||||||
|
|
||||||
this.globalEventServie.publishNoteStream(note.id, 'reacted', {
|
this.globalEventService.publishNoteStream(note.id, 'reacted', {
|
||||||
reaction: decodedReaction.reaction,
|
reaction: decodedReaction.reaction,
|
||||||
emoji: emoji != null ? {
|
emoji: emoji != null ? {
|
||||||
name: emoji.host ? `${emoji.name}@${emoji.host}` : `${emoji.name}@.`,
|
name: emoji.host ? `${emoji.name}@${emoji.host}` : `${emoji.name}@.`,
|
||||||
@@ -229,7 +228,7 @@ export class ReactionService {
|
|||||||
|
|
||||||
if (!user.isBot) this.notesRepository.decrement({ id: note.id }, 'score', 1);
|
if (!user.isBot) this.notesRepository.decrement({ id: note.id }, 'score', 1);
|
||||||
|
|
||||||
this.globalEventServie.publishNoteStream(note.id, 'unreacted', {
|
this.globalEventService.publishNoteStream(note.id, 'unreacted', {
|
||||||
reaction: this.decodeReaction(exist.reaction).reaction,
|
reaction: this.decodeReaction(exist.reaction).reaction,
|
||||||
userId: user.id,
|
userId: user.id,
|
||||||
});
|
});
|
||||||
|
@@ -91,10 +91,12 @@ export class RoleService implements OnApplicationShutdown {
|
|||||||
case 'roleCreated': {
|
case 'roleCreated': {
|
||||||
const cached = this.rolesCache.get(null);
|
const cached = this.rolesCache.get(null);
|
||||||
if (cached) {
|
if (cached) {
|
||||||
body.createdAt = new Date(body.createdAt);
|
cached.push({
|
||||||
body.updatedAt = new Date(body.updatedAt);
|
...body,
|
||||||
body.lastUsedAt = new Date(body.lastUsedAt);
|
createdAt: new Date(body.createdAt),
|
||||||
cached.push(body);
|
updatedAt: new Date(body.updatedAt),
|
||||||
|
lastUsedAt: new Date(body.lastUsedAt),
|
||||||
|
});
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@@ -103,10 +105,12 @@ export class RoleService implements OnApplicationShutdown {
|
|||||||
if (cached) {
|
if (cached) {
|
||||||
const i = cached.findIndex(x => x.id === body.id);
|
const i = cached.findIndex(x => x.id === body.id);
|
||||||
if (i > -1) {
|
if (i > -1) {
|
||||||
body.createdAt = new Date(body.createdAt);
|
cached[i] = {
|
||||||
body.updatedAt = new Date(body.updatedAt);
|
...body,
|
||||||
body.lastUsedAt = new Date(body.lastUsedAt);
|
createdAt: new Date(body.createdAt),
|
||||||
cached[i] = body;
|
updatedAt: new Date(body.updatedAt),
|
||||||
|
lastUsedAt: new Date(body.lastUsedAt),
|
||||||
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
@@ -121,8 +125,10 @@ export class RoleService implements OnApplicationShutdown {
|
|||||||
case 'userRoleAssigned': {
|
case 'userRoleAssigned': {
|
||||||
const cached = this.roleAssignmentByUserIdCache.get(body.userId);
|
const cached = this.roleAssignmentByUserIdCache.get(body.userId);
|
||||||
if (cached) {
|
if (cached) {
|
||||||
body.createdAt = new Date(body.createdAt);
|
cached.push({
|
||||||
cached.push(body);
|
...body,
|
||||||
|
createdAt: new Date(body.createdAt),
|
||||||
|
});
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@@ -196,6 +202,19 @@ export class RoleService implements OnApplicationShutdown {
|
|||||||
return [...assignedRoles, ...matchedCondRoles];
|
return [...assignedRoles, ...matchedCondRoles];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 指定ユーザーのバッジロール一覧取得
|
||||||
|
*/
|
||||||
|
@bindThis
|
||||||
|
public async getUserBadgeRoles(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 assignedBadgeRoles = roles.filter(r => r.asBadge && assignedRoleIds.includes(r.id));
|
||||||
|
// コンディショナルロールも含めるのは負荷高そうだから一旦無し
|
||||||
|
return assignedBadgeRoles;
|
||||||
|
}
|
||||||
|
|
||||||
@bindThis
|
@bindThis
|
||||||
public async getUserPolicies(userId: User['id'] | null): Promise<RolePolicies> {
|
public async getUserPolicies(userId: User['id'] | null): Promise<RolePolicies> {
|
||||||
const meta = await this.metaService.fetch();
|
const meta = await this.metaService.fetch();
|
||||||
|
@@ -33,7 +33,7 @@ export class S3Service {
|
|||||||
? false
|
? false
|
||||||
: meta.objectStorageS3ForcePathStyle,
|
: meta.objectStorageS3ForcePathStyle,
|
||||||
httpOptions: {
|
httpOptions: {
|
||||||
agent: this.httpRequestService.getHttpAgentByUrl(new URL(u), !meta.objectStorageUseProxy),
|
agent: this.httpRequestService.getAgentByUrl(new URL(u), !meta.objectStorageUseProxy),
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@@ -1,5 +1,6 @@
|
|||||||
|
|
||||||
import { Inject, Injectable } from '@nestjs/common';
|
import { Inject, Injectable, OnApplicationShutdown } from '@nestjs/common';
|
||||||
|
import Redis from 'ioredis';
|
||||||
import { IdService } from '@/core/IdService.js';
|
import { IdService } from '@/core/IdService.js';
|
||||||
import type { CacheableUser, User } from '@/models/entities/User.js';
|
import type { CacheableUser, User } from '@/models/entities/User.js';
|
||||||
import type { Blocking } from '@/models/entities/Blocking.js';
|
import type { Blocking } from '@/models/entities/Blocking.js';
|
||||||
@@ -7,7 +8,6 @@ import { QueueService } from '@/core/QueueService.js';
|
|||||||
import { GlobalEventService } from '@/core/GlobalEventService.js';
|
import { GlobalEventService } from '@/core/GlobalEventService.js';
|
||||||
import PerUserFollowingChart from '@/core/chart/charts/per-user-following.js';
|
import PerUserFollowingChart from '@/core/chart/charts/per-user-following.js';
|
||||||
import { DI } from '@/di-symbols.js';
|
import { DI } from '@/di-symbols.js';
|
||||||
import logger from '@/logger.js';
|
|
||||||
import type { UsersRepository, FollowingsRepository, FollowRequestsRepository, BlockingsRepository, UserListsRepository, UserListJoiningsRepository } from '@/models/index.js';
|
import type { UsersRepository, FollowingsRepository, FollowRequestsRepository, BlockingsRepository, UserListsRepository, UserListJoiningsRepository } from '@/models/index.js';
|
||||||
import Logger from '@/logger.js';
|
import Logger from '@/logger.js';
|
||||||
import { UserEntityService } from '@/core/entities/UserEntityService.js';
|
import { UserEntityService } from '@/core/entities/UserEntityService.js';
|
||||||
@@ -15,12 +15,20 @@ import { ApRendererService } from '@/core/activitypub/ApRendererService.js';
|
|||||||
import { LoggerService } from '@/core/LoggerService.js';
|
import { LoggerService } from '@/core/LoggerService.js';
|
||||||
import { WebhookService } from '@/core/WebhookService.js';
|
import { WebhookService } from '@/core/WebhookService.js';
|
||||||
import { bindThis } from '@/decorators.js';
|
import { bindThis } from '@/decorators.js';
|
||||||
|
import { Cache } from '@/misc/cache.js';
|
||||||
|
import { StreamMessages } from '@/server/api/stream/types.js';
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class UserBlockingService {
|
export class UserBlockingService implements OnApplicationShutdown {
|
||||||
private logger: Logger;
|
private logger: Logger;
|
||||||
|
|
||||||
|
// キーがユーザーIDで、値がそのユーザーがブロックしているユーザーのIDのリストなキャッシュ
|
||||||
|
private blockingsByUserIdCache: Cache<User['id'][]>;
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
|
@Inject(DI.redisSubscriber)
|
||||||
|
private redisSubscriber: Redis.Redis,
|
||||||
|
|
||||||
@Inject(DI.usersRepository)
|
@Inject(DI.usersRepository)
|
||||||
private usersRepository: UsersRepository,
|
private usersRepository: UsersRepository,
|
||||||
|
|
||||||
@@ -42,13 +50,44 @@ export class UserBlockingService {
|
|||||||
private userEntityService: UserEntityService,
|
private userEntityService: UserEntityService,
|
||||||
private idService: IdService,
|
private idService: IdService,
|
||||||
private queueService: QueueService,
|
private queueService: QueueService,
|
||||||
private globalEventServie: GlobalEventService,
|
private globalEventService: GlobalEventService,
|
||||||
private webhookService: WebhookService,
|
private webhookService: WebhookService,
|
||||||
private apRendererService: ApRendererService,
|
private apRendererService: ApRendererService,
|
||||||
private perUserFollowingChart: PerUserFollowingChart,
|
private perUserFollowingChart: PerUserFollowingChart,
|
||||||
private loggerService: LoggerService,
|
private loggerService: LoggerService,
|
||||||
) {
|
) {
|
||||||
this.logger = this.loggerService.getLogger('user-block');
|
this.logger = this.loggerService.getLogger('user-block');
|
||||||
|
|
||||||
|
this.blockingsByUserIdCache = new Cache<User['id'][]>(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 'blockingCreated': {
|
||||||
|
const cached = this.blockingsByUserIdCache.get(body.blockerId);
|
||||||
|
if (cached) {
|
||||||
|
this.blockingsByUserIdCache.set(body.blockerId, [...cached, ...[body.blockeeId]]);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case 'blockingDeleted': {
|
||||||
|
const cached = this.blockingsByUserIdCache.get(body.blockerId);
|
||||||
|
if (cached) {
|
||||||
|
this.blockingsByUserIdCache.set(body.blockerId, cached.filter(x => x !== body.blockeeId));
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@bindThis
|
@bindThis
|
||||||
@@ -72,6 +111,11 @@ export class UserBlockingService {
|
|||||||
|
|
||||||
await this.blockingsRepository.insert(blocking);
|
await this.blockingsRepository.insert(blocking);
|
||||||
|
|
||||||
|
this.globalEventService.publishInternalEvent('blockingCreated', {
|
||||||
|
blockerId: blocker.id,
|
||||||
|
blockeeId: blockee.id,
|
||||||
|
});
|
||||||
|
|
||||||
if (this.userEntityService.isLocalUser(blocker) && this.userEntityService.isRemoteUser(blockee)) {
|
if (this.userEntityService.isLocalUser(blocker) && this.userEntityService.isRemoteUser(blockee)) {
|
||||||
const content = this.apRendererService.renderActivity(this.apRendererService.renderBlock(blocking));
|
const content = this.apRendererService.renderActivity(this.apRendererService.renderBlock(blocking));
|
||||||
this.queueService.deliver(blocker, content, blockee.inbox);
|
this.queueService.deliver(blocker, content, blockee.inbox);
|
||||||
@@ -97,15 +141,15 @@ export class UserBlockingService {
|
|||||||
if (this.userEntityService.isLocalUser(followee)) {
|
if (this.userEntityService.isLocalUser(followee)) {
|
||||||
this.userEntityService.pack(followee, followee, {
|
this.userEntityService.pack(followee, followee, {
|
||||||
detail: true,
|
detail: true,
|
||||||
}).then(packed => this.globalEventServie.publishMainStream(followee.id, 'meUpdated', packed));
|
}).then(packed => this.globalEventService.publishMainStream(followee.id, 'meUpdated', packed));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this.userEntityService.isLocalUser(follower)) {
|
if (this.userEntityService.isLocalUser(follower)) {
|
||||||
this.userEntityService.pack(followee, follower, {
|
this.userEntityService.pack(followee, follower, {
|
||||||
detail: true,
|
detail: true,
|
||||||
}).then(async packed => {
|
}).then(async packed => {
|
||||||
this.globalEventServie.publishUserEvent(follower.id, 'unfollow', packed);
|
this.globalEventService.publishUserEvent(follower.id, 'unfollow', packed);
|
||||||
this.globalEventServie.publishMainStream(follower.id, 'unfollow', packed);
|
this.globalEventService.publishMainStream(follower.id, 'unfollow', packed);
|
||||||
|
|
||||||
const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === follower.id && x.on.includes('unfollow'));
|
const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === follower.id && x.on.includes('unfollow'));
|
||||||
for (const webhook of webhooks) {
|
for (const webhook of webhooks) {
|
||||||
@@ -152,8 +196,8 @@ export class UserBlockingService {
|
|||||||
this.userEntityService.pack(followee, follower, {
|
this.userEntityService.pack(followee, follower, {
|
||||||
detail: true,
|
detail: true,
|
||||||
}).then(async packed => {
|
}).then(async packed => {
|
||||||
this.globalEventServie.publishUserEvent(follower.id, 'unfollow', packed);
|
this.globalEventService.publishUserEvent(follower.id, 'unfollow', packed);
|
||||||
this.globalEventServie.publishMainStream(follower.id, 'unfollow', packed);
|
this.globalEventService.publishMainStream(follower.id, 'unfollow', packed);
|
||||||
|
|
||||||
const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === follower.id && x.on.includes('unfollow'));
|
const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === follower.id && x.on.includes('unfollow'));
|
||||||
for (const webhook of webhooks) {
|
for (const webhook of webhooks) {
|
||||||
@@ -210,10 +254,31 @@ export class UserBlockingService {
|
|||||||
|
|
||||||
await this.blockingsRepository.delete(blocking.id);
|
await this.blockingsRepository.delete(blocking.id);
|
||||||
|
|
||||||
|
this.globalEventService.publishInternalEvent('blockingDeleted', {
|
||||||
|
blockerId: blocker.id,
|
||||||
|
blockeeId: blockee.id,
|
||||||
|
});
|
||||||
|
|
||||||
// deliver if remote bloking
|
// deliver if remote bloking
|
||||||
if (this.userEntityService.isLocalUser(blocker) && this.userEntityService.isRemoteUser(blockee)) {
|
if (this.userEntityService.isLocalUser(blocker) && this.userEntityService.isRemoteUser(blockee)) {
|
||||||
const content = this.apRendererService.renderActivity(this.apRendererService.renderUndo(this.apRendererService.renderBlock(blocking), blocker));
|
const content = this.apRendererService.renderActivity(this.apRendererService.renderUndo(this.apRendererService.renderBlock(blocking), blocker));
|
||||||
this.queueService.deliver(blocker, content, blockee.inbox);
|
this.queueService.deliver(blocker, content, blockee.inbox);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@bindThis
|
||||||
|
public async checkBlocked(blockerId: User['id'], blockeeId: User['id']): Promise<boolean> {
|
||||||
|
const blockedUserIds = await this.blockingsByUserIdCache.fetch(blockerId, () => this.blockingsRepository.find({
|
||||||
|
where: {
|
||||||
|
blockerId,
|
||||||
|
},
|
||||||
|
select: ['blockeeId'],
|
||||||
|
}).then(records => records.map(record => record.blockeeId)));
|
||||||
|
return blockedUserIds.includes(blockeeId);
|
||||||
|
}
|
||||||
|
|
||||||
|
@bindThis
|
||||||
|
public onApplicationShutdown(signal?: string | undefined) {
|
||||||
|
this.redisSubscriber.off('message', this.onMessage);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@@ -12,10 +12,11 @@ import { FederatedInstanceService } from '@/core/FederatedInstanceService.js';
|
|||||||
import { WebhookService } from '@/core/WebhookService.js';
|
import { WebhookService } from '@/core/WebhookService.js';
|
||||||
import { CreateNotificationService } from '@/core/CreateNotificationService.js';
|
import { CreateNotificationService } from '@/core/CreateNotificationService.js';
|
||||||
import { DI } from '@/di-symbols.js';
|
import { DI } from '@/di-symbols.js';
|
||||||
import type { BlockingsRepository, FollowingsRepository, FollowRequestsRepository, InstancesRepository, UserProfilesRepository, UsersRepository } from '@/models/index.js';
|
import type { FollowingsRepository, FollowRequestsRepository, InstancesRepository, UserProfilesRepository, UsersRepository } from '@/models/index.js';
|
||||||
import { UserEntityService } from '@/core/entities/UserEntityService.js';
|
import { UserEntityService } from '@/core/entities/UserEntityService.js';
|
||||||
import { ApRendererService } from '@/core/activitypub/ApRendererService.js';
|
import { ApRendererService } from '@/core/activitypub/ApRendererService.js';
|
||||||
import { bindThis } from '@/decorators.js';
|
import { bindThis } from '@/decorators.js';
|
||||||
|
import { UserBlockingService } from '@/core/UserBlockingService.js';
|
||||||
import Logger from '../logger.js';
|
import Logger from '../logger.js';
|
||||||
|
|
||||||
const logger = new Logger('following/create');
|
const logger = new Logger('following/create');
|
||||||
@@ -48,21 +49,18 @@ export class UserFollowingService {
|
|||||||
@Inject(DI.followRequestsRepository)
|
@Inject(DI.followRequestsRepository)
|
||||||
private followRequestsRepository: FollowRequestsRepository,
|
private followRequestsRepository: FollowRequestsRepository,
|
||||||
|
|
||||||
@Inject(DI.blockingsRepository)
|
|
||||||
private blockingsRepository: BlockingsRepository,
|
|
||||||
|
|
||||||
@Inject(DI.instancesRepository)
|
@Inject(DI.instancesRepository)
|
||||||
private instancesRepository: InstancesRepository,
|
private instancesRepository: InstancesRepository,
|
||||||
|
|
||||||
private userEntityService: UserEntityService,
|
private userEntityService: UserEntityService,
|
||||||
|
private userBlockingService: UserBlockingService,
|
||||||
private idService: IdService,
|
private idService: IdService,
|
||||||
private queueService: QueueService,
|
private queueService: QueueService,
|
||||||
private globalEventServie: GlobalEventService,
|
private globalEventService: GlobalEventService,
|
||||||
private createNotificationService: CreateNotificationService,
|
private createNotificationService: CreateNotificationService,
|
||||||
private federatedInstanceService: FederatedInstanceService,
|
private federatedInstanceService: FederatedInstanceService,
|
||||||
private webhookService: WebhookService,
|
private webhookService: WebhookService,
|
||||||
private apRendererService: ApRendererService,
|
private apRendererService: ApRendererService,
|
||||||
private globalEventService: GlobalEventService,
|
|
||||||
private perUserFollowingChart: PerUserFollowingChart,
|
private perUserFollowingChart: PerUserFollowingChart,
|
||||||
private instanceChart: InstanceChart,
|
private instanceChart: InstanceChart,
|
||||||
) {
|
) {
|
||||||
@@ -77,28 +75,22 @@ export class UserFollowingService {
|
|||||||
|
|
||||||
// check blocking
|
// check blocking
|
||||||
const [blocking, blocked] = await Promise.all([
|
const [blocking, blocked] = await Promise.all([
|
||||||
this.blockingsRepository.findOneBy({
|
this.userBlockingService.checkBlocked(follower.id, followee.id),
|
||||||
blockerId: follower.id,
|
this.userBlockingService.checkBlocked(followee.id, follower.id),
|
||||||
blockeeId: followee.id,
|
|
||||||
}),
|
|
||||||
this.blockingsRepository.findOneBy({
|
|
||||||
blockerId: followee.id,
|
|
||||||
blockeeId: follower.id,
|
|
||||||
}),
|
|
||||||
]);
|
]);
|
||||||
|
|
||||||
if (this.userEntityService.isRemoteUser(follower) && this.userEntityService.isLocalUser(followee) && blocked) {
|
if (this.userEntityService.isRemoteUser(follower) && this.userEntityService.isLocalUser(followee) && blocked) {
|
||||||
// リモートフォローを受けてブロックしていた場合は、エラーにするのではなくRejectを送り返しておしまい。
|
// リモートフォローを受けてブロックしていた場合は、エラーにするのではなくRejectを送り返しておしまい。
|
||||||
const content = this.apRendererService.renderActivity(this.apRendererService.renderReject(this.apRendererService.renderFollow(follower, followee, requestId), followee));
|
const content = this.apRendererService.renderActivity(this.apRendererService.renderReject(this.apRendererService.renderFollow(follower, followee, requestId), followee));
|
||||||
this.queueService.deliver(followee, content, follower.inbox);
|
this.queueService.deliver(followee, content, follower.inbox);
|
||||||
return;
|
return;
|
||||||
} else if (this.userEntityService.isRemoteUser(follower) && this.userEntityService.isLocalUser(followee) && blocking) {
|
} else if (this.userEntityService.isRemoteUser(follower) && this.userEntityService.isLocalUser(followee) && blocking) {
|
||||||
// リモートフォローを受けてブロックされているはずの場合だったら、ブロック解除しておく。
|
// リモートフォローを受けてブロックされているはずの場合だったら、ブロック解除しておく。
|
||||||
await this.blockingsRepository.delete(blocking.id);
|
await this.userBlockingService.unblock(follower, followee);
|
||||||
} else {
|
} else {
|
||||||
// それ以外は単純に例外
|
// それ以外は単純に例外
|
||||||
if (blocking != null) throw new IdentifiableError('710e8fb0-b8c3-4922-be49-d5d93d8e6a6e', 'blocking');
|
if (blocking) throw new IdentifiableError('710e8fb0-b8c3-4922-be49-d5d93d8e6a6e', 'blocking');
|
||||||
if (blocked != null) throw new IdentifiableError('3338392a-f764-498d-8855-db939dcf8c48', 'blocked');
|
if (blocked) throw new IdentifiableError('3338392a-f764-498d-8855-db939dcf8c48', 'blocked');
|
||||||
}
|
}
|
||||||
|
|
||||||
const followeeProfile = await this.userProfilesRepository.findOneByOrFail({ userId: followee.id });
|
const followeeProfile = await this.userProfilesRepository.findOneByOrFail({ userId: followee.id });
|
||||||
@@ -227,8 +219,8 @@ export class UserFollowingService {
|
|||||||
this.userEntityService.pack(followee.id, follower, {
|
this.userEntityService.pack(followee.id, follower, {
|
||||||
detail: true,
|
detail: true,
|
||||||
}).then(async packed => {
|
}).then(async packed => {
|
||||||
this.globalEventServie.publishUserEvent(follower.id, 'follow', packed as Packed<'UserDetailedNotMe'>);
|
this.globalEventService.publishUserEvent(follower.id, 'follow', packed as Packed<'UserDetailedNotMe'>);
|
||||||
this.globalEventServie.publishMainStream(follower.id, 'follow', packed as Packed<'UserDetailedNotMe'>);
|
this.globalEventService.publishMainStream(follower.id, 'follow', packed as Packed<'UserDetailedNotMe'>);
|
||||||
|
|
||||||
const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === follower.id && x.on.includes('follow'));
|
const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === follower.id && x.on.includes('follow'));
|
||||||
for (const webhook of webhooks) {
|
for (const webhook of webhooks) {
|
||||||
@@ -242,7 +234,7 @@ export class UserFollowingService {
|
|||||||
// Publish followed event
|
// Publish followed event
|
||||||
if (this.userEntityService.isLocalUser(followee)) {
|
if (this.userEntityService.isLocalUser(followee)) {
|
||||||
this.userEntityService.pack(follower.id, followee).then(async packed => {
|
this.userEntityService.pack(follower.id, followee).then(async packed => {
|
||||||
this.globalEventServie.publishMainStream(followee.id, 'followed', packed);
|
this.globalEventService.publishMainStream(followee.id, 'followed', packed);
|
||||||
|
|
||||||
const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === followee.id && x.on.includes('followed'));
|
const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === followee.id && x.on.includes('followed'));
|
||||||
for (const webhook of webhooks) {
|
for (const webhook of webhooks) {
|
||||||
@@ -288,8 +280,8 @@ export class UserFollowingService {
|
|||||||
this.userEntityService.pack(followee.id, follower, {
|
this.userEntityService.pack(followee.id, follower, {
|
||||||
detail: true,
|
detail: true,
|
||||||
}).then(async packed => {
|
}).then(async packed => {
|
||||||
this.globalEventServie.publishUserEvent(follower.id, 'unfollow', packed);
|
this.globalEventService.publishUserEvent(follower.id, 'unfollow', packed);
|
||||||
this.globalEventServie.publishMainStream(follower.id, 'unfollow', packed);
|
this.globalEventService.publishMainStream(follower.id, 'unfollow', packed);
|
||||||
|
|
||||||
const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === follower.id && x.on.includes('unfollow'));
|
const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === follower.id && x.on.includes('unfollow'));
|
||||||
for (const webhook of webhooks) {
|
for (const webhook of webhooks) {
|
||||||
@@ -357,18 +349,12 @@ export class UserFollowingService {
|
|||||||
|
|
||||||
// check blocking
|
// check blocking
|
||||||
const [blocking, blocked] = await Promise.all([
|
const [blocking, blocked] = await Promise.all([
|
||||||
this.blockingsRepository.findOneBy({
|
this.userBlockingService.checkBlocked(follower.id, followee.id),
|
||||||
blockerId: follower.id,
|
this.userBlockingService.checkBlocked(followee.id, follower.id),
|
||||||
blockeeId: followee.id,
|
|
||||||
}),
|
|
||||||
this.blockingsRepository.findOneBy({
|
|
||||||
blockerId: followee.id,
|
|
||||||
blockeeId: follower.id,
|
|
||||||
}),
|
|
||||||
]);
|
]);
|
||||||
|
|
||||||
if (blocking != null) throw new Error('blocking');
|
if (blocking) throw new Error('blocking');
|
||||||
if (blocked != null) throw new Error('blocked');
|
if (blocked) throw new Error('blocked');
|
||||||
|
|
||||||
const followRequest = await this.followRequestsRepository.insert({
|
const followRequest = await this.followRequestsRepository.insert({
|
||||||
id: this.idService.genId(),
|
id: this.idService.genId(),
|
||||||
@@ -388,11 +374,11 @@ export class UserFollowingService {
|
|||||||
|
|
||||||
// Publish receiveRequest event
|
// Publish receiveRequest event
|
||||||
if (this.userEntityService.isLocalUser(followee)) {
|
if (this.userEntityService.isLocalUser(followee)) {
|
||||||
this.userEntityService.pack(follower.id, followee).then(packed => this.globalEventServie.publishMainStream(followee.id, 'receiveFollowRequest', packed));
|
this.userEntityService.pack(follower.id, followee).then(packed => this.globalEventService.publishMainStream(followee.id, 'receiveFollowRequest', packed));
|
||||||
|
|
||||||
this.userEntityService.pack(followee.id, followee, {
|
this.userEntityService.pack(followee.id, followee, {
|
||||||
detail: true,
|
detail: true,
|
||||||
}).then(packed => this.globalEventServie.publishMainStream(followee.id, 'meUpdated', packed));
|
}).then(packed => this.globalEventService.publishMainStream(followee.id, 'meUpdated', packed));
|
||||||
|
|
||||||
// 通知を作成
|
// 通知を作成
|
||||||
this.createNotificationService.createNotification(followee.id, 'receiveFollowRequest', {
|
this.createNotificationService.createNotification(followee.id, 'receiveFollowRequest', {
|
||||||
@@ -440,7 +426,7 @@ export class UserFollowingService {
|
|||||||
|
|
||||||
this.userEntityService.pack(followee.id, followee, {
|
this.userEntityService.pack(followee.id, followee, {
|
||||||
detail: true,
|
detail: true,
|
||||||
}).then(packed => this.globalEventServie.publishMainStream(followee.id, 'meUpdated', packed));
|
}).then(packed => this.globalEventService.publishMainStream(followee.id, 'meUpdated', packed));
|
||||||
}
|
}
|
||||||
|
|
||||||
@bindThis
|
@bindThis
|
||||||
@@ -468,7 +454,7 @@ export class UserFollowingService {
|
|||||||
|
|
||||||
this.userEntityService.pack(followee.id, followee, {
|
this.userEntityService.pack(followee.id, followee, {
|
||||||
detail: true,
|
detail: true,
|
||||||
}).then(packed => this.globalEventServie.publishMainStream(followee.id, 'meUpdated', packed));
|
}).then(packed => this.globalEventService.publishMainStream(followee.id, 'meUpdated', packed));
|
||||||
}
|
}
|
||||||
|
|
||||||
@bindThis
|
@bindThis
|
||||||
@@ -583,8 +569,8 @@ export class UserFollowingService {
|
|||||||
detail: true,
|
detail: true,
|
||||||
});
|
});
|
||||||
|
|
||||||
this.globalEventServie.publishUserEvent(follower.id, 'unfollow', packedFollowee);
|
this.globalEventService.publishUserEvent(follower.id, 'unfollow', packedFollowee);
|
||||||
this.globalEventServie.publishMainStream(follower.id, 'unfollow', packedFollowee);
|
this.globalEventService.publishMainStream(follower.id, 'unfollow', packedFollowee);
|
||||||
|
|
||||||
const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === follower.id && x.on.includes('unfollow'));
|
const webhooks = (await this.webhookService.getActiveWebhooks()).filter(x => x.userId === follower.id && x.on.includes('unfollow'));
|
||||||
for (const webhook of webhooks) {
|
for (const webhook of webhooks) {
|
||||||
|
@@ -25,7 +25,7 @@ export class UserListService {
|
|||||||
private idService: IdService,
|
private idService: IdService,
|
||||||
private userFollowingService: UserFollowingService,
|
private userFollowingService: UserFollowingService,
|
||||||
private roleService: RoleService,
|
private roleService: RoleService,
|
||||||
private globalEventServie: GlobalEventService,
|
private globalEventService: GlobalEventService,
|
||||||
private proxyAccountService: ProxyAccountService,
|
private proxyAccountService: ProxyAccountService,
|
||||||
) {
|
) {
|
||||||
}
|
}
|
||||||
@@ -46,7 +46,7 @@ export class UserListService {
|
|||||||
userListId: list.id,
|
userListId: list.id,
|
||||||
} as UserListJoining);
|
} as UserListJoining);
|
||||||
|
|
||||||
this.globalEventServie.publishUserListStream(list.id, 'userAdded', await this.userEntityService.pack(target));
|
this.globalEventService.publishUserListStream(list.id, 'userAdded', await this.userEntityService.pack(target));
|
||||||
|
|
||||||
// このインスタンス内にこのリモートユーザーをフォローしているユーザーがいなくても投稿を受け取るためにダミーのユーザーがフォローしたということにする
|
// このインスタンス内にこのリモートユーザーをフォローしているユーザーがいなくても投稿を受け取るためにダミーのユーザーがフォローしたということにする
|
||||||
if (this.userEntityService.isRemoteUser(target)) {
|
if (this.userEntityService.isRemoteUser(target)) {
|
||||||
|
@@ -18,7 +18,7 @@ export class UserMutingService {
|
|||||||
|
|
||||||
private idService: IdService,
|
private idService: IdService,
|
||||||
private queueService: QueueService,
|
private queueService: QueueService,
|
||||||
private globalEventServie: GlobalEventService,
|
private globalEventService: GlobalEventService,
|
||||||
) {
|
) {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -44,16 +44,25 @@ export class WebhookService implements OnApplicationShutdown {
|
|||||||
switch (type) {
|
switch (type) {
|
||||||
case 'webhookCreated':
|
case 'webhookCreated':
|
||||||
if (body.active) {
|
if (body.active) {
|
||||||
this.webhooks.push(body);
|
this.webhooks.push({
|
||||||
|
...body,
|
||||||
|
createdAt: new Date(body.createdAt),
|
||||||
|
});
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case 'webhookUpdated':
|
case 'webhookUpdated':
|
||||||
if (body.active) {
|
if (body.active) {
|
||||||
const i = this.webhooks.findIndex(a => a.id === body.id);
|
const i = this.webhooks.findIndex(a => a.id === body.id);
|
||||||
if (i > -1) {
|
if (i > -1) {
|
||||||
this.webhooks[i] = body;
|
this.webhooks[i] = {
|
||||||
|
...body,
|
||||||
|
createdAt: new Date(body.createdAt),
|
||||||
|
};
|
||||||
} else {
|
} else {
|
||||||
this.webhooks.push(body);
|
this.webhooks.push({
|
||||||
|
...body,
|
||||||
|
createdAt: new Date(body.createdAt),
|
||||||
|
});
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
this.webhooks = this.webhooks.filter(a => a.id !== body.id);
|
this.webhooks = this.webhooks.filter(a => a.id !== body.id);
|
||||||
|
@@ -21,11 +21,11 @@ import { UserEntityService } from '@/core/entities/UserEntityService.js';
|
|||||||
import { DriveFileEntityService } from '@/core/entities/DriveFileEntityService.js';
|
import { DriveFileEntityService } from '@/core/entities/DriveFileEntityService.js';
|
||||||
import type { UserKeypair } from '@/models/entities/UserKeypair.js';
|
import type { UserKeypair } from '@/models/entities/UserKeypair.js';
|
||||||
import type { UsersRepository, UserProfilesRepository, NotesRepository, DriveFilesRepository, EmojisRepository, PollsRepository } from '@/models/index.js';
|
import type { UsersRepository, UserProfilesRepository, NotesRepository, DriveFilesRepository, EmojisRepository, PollsRepository } from '@/models/index.js';
|
||||||
|
import { bindThis } from '@/decorators.js';
|
||||||
import { LdSignatureService } from './LdSignatureService.js';
|
import { LdSignatureService } from './LdSignatureService.js';
|
||||||
import { ApMfmService } from './ApMfmService.js';
|
import { ApMfmService } from './ApMfmService.js';
|
||||||
import type { IActivity, IObject } from './type.js';
|
import type { IActivity, IObject } from './type.js';
|
||||||
import type { IIdentifier } from './models/identifier.js';
|
import type { IIdentifier } from './models/identifier.js';
|
||||||
import { bindThis } from '@/decorators.js';
|
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class ApRendererService {
|
export class ApRendererService {
|
||||||
@@ -274,7 +274,7 @@ export class ApRendererService {
|
|||||||
} as any;
|
} as any;
|
||||||
|
|
||||||
if (reaction.startsWith(':')) {
|
if (reaction.startsWith(':')) {
|
||||||
const name = reaction.replace(/:/g, '');
|
const name = reaction.replaceAll(':', '');
|
||||||
const emoji = await this.emojisRepository.findOneBy({
|
const emoji = await this.emojisRepository.findOneBy({
|
||||||
name,
|
name,
|
||||||
host: IsNull(),
|
host: IsNull(),
|
||||||
|
@@ -5,7 +5,7 @@ import { DI } from '@/di-symbols.js';
|
|||||||
import type { Config } from '@/config.js';
|
import type { Config } from '@/config.js';
|
||||||
import type { User } from '@/models/entities/User.js';
|
import type { User } from '@/models/entities/User.js';
|
||||||
import { UserKeypairStoreService } from '@/core/UserKeypairStoreService.js';
|
import { UserKeypairStoreService } from '@/core/UserKeypairStoreService.js';
|
||||||
import { HttpRequestService, UndiciFetcher } from '@/core/HttpRequestService.js';
|
import { HttpRequestService } from '@/core/HttpRequestService.js';
|
||||||
import { LoggerService } from '@/core/LoggerService.js';
|
import { LoggerService } from '@/core/LoggerService.js';
|
||||||
import { bindThis } from '@/decorators.js';
|
import { bindThis } from '@/decorators.js';
|
||||||
import type Logger from '@/logger.js';
|
import type Logger from '@/logger.js';
|
||||||
@@ -30,7 +30,6 @@ type PrivateKey = {
|
|||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class ApRequestService {
|
export class ApRequestService {
|
||||||
private undiciFetcher: UndiciFetcher;
|
|
||||||
private logger: Logger;
|
private logger: Logger;
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
@@ -41,10 +40,8 @@ export class ApRequestService {
|
|||||||
private httpRequestService: HttpRequestService,
|
private httpRequestService: HttpRequestService,
|
||||||
private loggerService: LoggerService,
|
private loggerService: LoggerService,
|
||||||
) {
|
) {
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
|
||||||
this.logger = this.loggerService?.getLogger('ap-request'); // なぜか TypeError: Cannot read properties of undefined (reading 'getLogger') と言われる
|
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
|
@bindThis
|
||||||
@@ -163,14 +160,11 @@ export class ApRequestService {
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
await this.undiciFetcher.fetch(
|
await this.httpRequestService.send(url, {
|
||||||
url,
|
method: req.request.method,
|
||||||
{
|
headers: req.request.headers,
|
||||||
method: req.request.method,
|
body,
|
||||||
headers: req.request.headers,
|
});
|
||||||
body,
|
|
||||||
}
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -192,13 +186,10 @@ export class ApRequestService {
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const res = await this.httpRequestService.fetch(
|
const res = await this.httpRequestService.send(url, {
|
||||||
url,
|
method: req.request.method,
|
||||||
{
|
headers: req.request.headers,
|
||||||
method: req.request.method,
|
});
|
||||||
headers: req.request.headers,
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
return await res.json();
|
return await res.json();
|
||||||
}
|
}
|
||||||
|
@@ -4,22 +4,21 @@ import { InstanceActorService } from '@/core/InstanceActorService.js';
|
|||||||
import type { NotesRepository, PollsRepository, NoteReactionsRepository, UsersRepository } from '@/models/index.js';
|
import type { NotesRepository, PollsRepository, NoteReactionsRepository, UsersRepository } from '@/models/index.js';
|
||||||
import type { Config } from '@/config.js';
|
import type { Config } from '@/config.js';
|
||||||
import { MetaService } from '@/core/MetaService.js';
|
import { MetaService } from '@/core/MetaService.js';
|
||||||
import { HttpRequestService, UndiciFetcher } from '@/core/HttpRequestService.js';
|
import { HttpRequestService } from '@/core/HttpRequestService.js';
|
||||||
import { DI } from '@/di-symbols.js';
|
import { DI } from '@/di-symbols.js';
|
||||||
import { UtilityService } from '@/core/UtilityService.js';
|
import { UtilityService } from '@/core/UtilityService.js';
|
||||||
import { bindThis } from '@/decorators.js';
|
import { bindThis } from '@/decorators.js';
|
||||||
|
import { LoggerService } from '@/core/LoggerService.js';
|
||||||
|
import type Logger from '@/logger.js';
|
||||||
import { isCollectionOrOrderedCollection } from './type.js';
|
import { isCollectionOrOrderedCollection } from './type.js';
|
||||||
import { ApDbResolverService } from './ApDbResolverService.js';
|
import { ApDbResolverService } from './ApDbResolverService.js';
|
||||||
import { ApRendererService } from './ApRendererService.js';
|
import { ApRendererService } from './ApRendererService.js';
|
||||||
import { ApRequestService } from './ApRequestService.js';
|
import { ApRequestService } from './ApRequestService.js';
|
||||||
import { LoggerService } from '@/core/LoggerService.js';
|
|
||||||
import type { IObject, ICollection, IOrderedCollection } from './type.js';
|
import type { IObject, ICollection, IOrderedCollection } from './type.js';
|
||||||
import type Logger from '@/logger.js';
|
|
||||||
|
|
||||||
export class Resolver {
|
export class Resolver {
|
||||||
private history: Set<string>;
|
private history: Set<string>;
|
||||||
private user?: ILocalUser;
|
private user?: ILocalUser;
|
||||||
private undiciFetcher: UndiciFetcher;
|
|
||||||
private logger: Logger;
|
private logger: Logger;
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
@@ -39,10 +38,8 @@ export class Resolver {
|
|||||||
private recursionLimit = 100,
|
private recursionLimit = 100,
|
||||||
) {
|
) {
|
||||||
this.history = new Set();
|
this.history = new Set();
|
||||||
this.logger = this.loggerService?.getLogger('ap-resolve'); // なぜか TypeError: Cannot read properties of undefined (reading 'getLogger') と言われる
|
// eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
|
||||||
this.undiciFetcher = new UndiciFetcher(this.httpRequestService.getStandardUndiciFetcherOption({
|
this.logger = this.loggerService?.getLogger('ap-resolve'); // なぜか TypeError: Cannot read properties of undefined (reading 'getLogger') と言われる
|
||||||
maxRedirections: 0,
|
|
||||||
}), this.logger);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@bindThis
|
@bindThis
|
||||||
@@ -106,7 +103,7 @@ export class Resolver {
|
|||||||
|
|
||||||
const object = (this.user
|
const object = (this.user
|
||||||
? await this.apRequestService.signedGet(value, this.user) as IObject
|
? await this.apRequestService.signedGet(value, this.user) as IObject
|
||||||
: await this.undiciFetcher.getJson<IObject>(value, 'application/activity+json, application/ld+json'));
|
: await this.httpRequestService.getJson(value, 'application/activity+json, application/ld+json')) as IObject;
|
||||||
|
|
||||||
if (object == null || (
|
if (object == null || (
|
||||||
Array.isArray(object['@context']) ?
|
Array.isArray(object['@context']) ?
|
||||||
|
@@ -1,5 +1,6 @@
|
|||||||
import * as crypto from 'node:crypto';
|
import * as crypto from 'node:crypto';
|
||||||
import { Inject, Injectable } from '@nestjs/common';
|
import { Inject, Injectable } from '@nestjs/common';
|
||||||
|
import jsonld from 'jsonld';
|
||||||
import { HttpRequestService } from '@/core/HttpRequestService.js';
|
import { HttpRequestService } from '@/core/HttpRequestService.js';
|
||||||
import { bindThis } from '@/decorators.js';
|
import { bindThis } from '@/decorators.js';
|
||||||
import { CONTEXTS } from './misc/contexts.js';
|
import { CONTEXTS } from './misc/contexts.js';
|
||||||
@@ -9,7 +10,7 @@ import { CONTEXTS } from './misc/contexts.js';
|
|||||||
class LdSignature {
|
class LdSignature {
|
||||||
public debug = false;
|
public debug = false;
|
||||||
public preLoad = true;
|
public preLoad = true;
|
||||||
public loderTimeout = 10 * 1000;
|
public loderTimeout = 5000;
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
private httpRequestService: HttpRequestService,
|
private httpRequestService: HttpRequestService,
|
||||||
@@ -84,7 +85,9 @@ class LdSignature {
|
|||||||
@bindThis
|
@bindThis
|
||||||
public async normalize(data: any) {
|
public async normalize(data: any) {
|
||||||
const customLoader = this.getLoader();
|
const customLoader = this.getLoader();
|
||||||
return 42;
|
return await jsonld.normalize(data, {
|
||||||
|
documentLoader: customLoader,
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@bindThis
|
@bindThis
|
||||||
@@ -115,19 +118,12 @@ class LdSignature {
|
|||||||
|
|
||||||
@bindThis
|
@bindThis
|
||||||
private async fetchDocument(url: string) {
|
private async fetchDocument(url: string) {
|
||||||
const json = await this.httpRequestService.fetch(
|
const json = await this.httpRequestService.send(url, {
|
||||||
url,
|
headers: {
|
||||||
{
|
Accept: 'application/ld+json, application/json',
|
||||||
headers: {
|
|
||||||
Accept: 'application/ld+json, application/json',
|
|
||||||
},
|
|
||||||
// TODO
|
|
||||||
//timeout: this.loderTimeout,
|
|
||||||
},
|
},
|
||||||
{
|
timeout: this.loderTimeout,
|
||||||
noOkError: true,
|
}, { throwErrorWhenResponseNotOk: false }).then(res => {
|
||||||
}
|
|
||||||
).then(res => {
|
|
||||||
if (!res.ok) {
|
if (!res.ok) {
|
||||||
throw `${res.status} ${res.statusText}`;
|
throw `${res.status} ${res.statusText}`;
|
||||||
} else {
|
} else {
|
||||||
|
@@ -48,6 +48,10 @@ export class ApImageService {
|
|||||||
throw new Error('invalid image: url not privided');
|
throw new Error('invalid image: url not privided');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!image.url.startsWith('https://')) {
|
||||||
|
throw new Error('invalid image: unexpected shcema of url: ' + image.url);
|
||||||
|
}
|
||||||
|
|
||||||
this.logger.info(`Creating the Image: ${image.url}`);
|
this.logger.info(`Creating the Image: ${image.url}`);
|
||||||
|
|
||||||
const instance = await this.metaService.fetch();
|
const instance = await this.metaService.fetch();
|
||||||
|
@@ -1,8 +1,7 @@
|
|||||||
import { forwardRef, Inject, Injectable } from '@nestjs/common';
|
import { forwardRef, Inject, Injectable } from '@nestjs/common';
|
||||||
import promiseLimit from 'promise-limit';
|
import promiseLimit from 'promise-limit';
|
||||||
import { DI } from '@/di-symbols.js';
|
import { DI } from '@/di-symbols.js';
|
||||||
import type { MessagingMessagesRepository, PollsRepository, EmojisRepository } from '@/models/index.js';
|
import type { MessagingMessagesRepository, PollsRepository, EmojisRepository, UsersRepository } from '@/models/index.js';
|
||||||
import type { UsersRepository } from '@/models/index.js';
|
|
||||||
import type { Config } from '@/config.js';
|
import type { Config } from '@/config.js';
|
||||||
import type { CacheableRemoteUser } from '@/models/entities/User.js';
|
import type { CacheableRemoteUser } from '@/models/entities/User.js';
|
||||||
import type { Note } from '@/models/entities/Note.js';
|
import type { Note } from '@/models/entities/Note.js';
|
||||||
@@ -18,6 +17,7 @@ import { PollService } from '@/core/PollService.js';
|
|||||||
import { StatusError } from '@/misc/status-error.js';
|
import { StatusError } from '@/misc/status-error.js';
|
||||||
import { UtilityService } from '@/core/UtilityService.js';
|
import { UtilityService } from '@/core/UtilityService.js';
|
||||||
import { MessagingService } from '@/core/MessagingService.js';
|
import { MessagingService } from '@/core/MessagingService.js';
|
||||||
|
import { bindThis } from '@/decorators.js';
|
||||||
import { getOneApId, getApId, getOneApHrefNullable, validPost, isEmoji, getApType } from '../type.js';
|
import { getOneApId, getApId, getOneApHrefNullable, validPost, isEmoji, getApType } from '../type.js';
|
||||||
// eslint-disable-next-line @typescript-eslint/consistent-type-imports
|
// eslint-disable-next-line @typescript-eslint/consistent-type-imports
|
||||||
import { ApLoggerService } from '../ApLoggerService.js';
|
import { ApLoggerService } from '../ApLoggerService.js';
|
||||||
@@ -32,7 +32,6 @@ import { ApQuestionService } from './ApQuestionService.js';
|
|||||||
import { ApImageService } from './ApImageService.js';
|
import { ApImageService } from './ApImageService.js';
|
||||||
import type { Resolver } from '../ApResolverService.js';
|
import type { Resolver } from '../ApResolverService.js';
|
||||||
import type { IObject, IPost } from '../type.js';
|
import type { IObject, IPost } from '../type.js';
|
||||||
import { bindThis } from '@/decorators.js';
|
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class ApNoteService {
|
export class ApNoteService {
|
||||||
@@ -133,6 +132,16 @@ export class ApNoteService {
|
|||||||
const note: IPost = object;
|
const note: IPost = object;
|
||||||
|
|
||||||
this.logger.debug(`Note fetched: ${JSON.stringify(note, null, 2)}`);
|
this.logger.debug(`Note fetched: ${JSON.stringify(note, null, 2)}`);
|
||||||
|
|
||||||
|
if (note.id && !note.id.startsWith('https://')) {
|
||||||
|
throw new Error('unexpected shcema of note.id: ' + note.id);
|
||||||
|
}
|
||||||
|
|
||||||
|
const url = getOneApHrefNullable(note.url);
|
||||||
|
|
||||||
|
if (url && !url.startsWith('https://')) {
|
||||||
|
throw new Error('unexpected shcema of note url: ' + url);
|
||||||
|
}
|
||||||
|
|
||||||
this.logger.info(`Creating the Note: ${note.id}`);
|
this.logger.info(`Creating the Note: ${note.id}`);
|
||||||
|
|
||||||
@@ -307,7 +316,7 @@ export class ApNoteService {
|
|||||||
apEmojis,
|
apEmojis,
|
||||||
poll,
|
poll,
|
||||||
uri: note.id,
|
uri: note.id,
|
||||||
url: getOneApHrefNullable(note.url),
|
url: url,
|
||||||
}, silent);
|
}, silent);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -29,6 +29,7 @@ import { UserNotePining } from '@/models/entities/UserNotePining.js';
|
|||||||
import { StatusError } from '@/misc/status-error.js';
|
import { StatusError } from '@/misc/status-error.js';
|
||||||
import type { UtilityService } from '@/core/UtilityService.js';
|
import type { UtilityService } from '@/core/UtilityService.js';
|
||||||
import type { UserEntityService } from '@/core/entities/UserEntityService.js';
|
import type { UserEntityService } from '@/core/entities/UserEntityService.js';
|
||||||
|
import { bindThis } from '@/decorators.js';
|
||||||
import { getApId, getApType, getOneApHrefNullable, isActor, isCollection, isCollectionOrOrderedCollection, isPropertyValue } from '../type.js';
|
import { getApId, getApType, getOneApHrefNullable, isActor, isCollection, isCollectionOrOrderedCollection, isPropertyValue } from '../type.js';
|
||||||
import { extractApHashtags } from './tag.js';
|
import { extractApHashtags } from './tag.js';
|
||||||
import type { OnModuleInit } from '@nestjs/common';
|
import type { OnModuleInit } from '@nestjs/common';
|
||||||
@@ -43,37 +44,6 @@ import type { IActor, IObject, IApPropertyValue } from '../type.js';
|
|||||||
const nameLength = 128;
|
const nameLength = 128;
|
||||||
const summaryLength = 2048;
|
const summaryLength = 2048;
|
||||||
|
|
||||||
const services: {
|
|
||||||
[x: string]: (id: string, username: string) => any
|
|
||||||
} = {
|
|
||||||
'misskey:authentication:twitter': (userId, screenName) => ({ userId, screenName }),
|
|
||||||
'misskey:authentication:github': (id, login) => ({ id, login }),
|
|
||||||
'misskey:authentication:discord': (id, name) => $discord(id, name),
|
|
||||||
};
|
|
||||||
|
|
||||||
const $discord = (id: string, name: string) => {
|
|
||||||
if (typeof name !== 'string') {
|
|
||||||
name = 'unknown#0000';
|
|
||||||
}
|
|
||||||
const [username, discriminator] = name.split('#');
|
|
||||||
return { id, username, discriminator };
|
|
||||||
};
|
|
||||||
|
|
||||||
function addService(target: { [x: string]: any }, source: IApPropertyValue) {
|
|
||||||
const service = services[source.name];
|
|
||||||
|
|
||||||
if (typeof source.value !== 'string') {
|
|
||||||
source.value = 'unknown';
|
|
||||||
}
|
|
||||||
|
|
||||||
const [id, username] = source.value.split('@');
|
|
||||||
|
|
||||||
if (service) {
|
|
||||||
target[source.name.split(':')[2]] = service(id, username);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
import { bindThis } from '@/decorators.js';
|
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class ApPersonService implements OnModuleInit {
|
export class ApPersonService implements OnModuleInit {
|
||||||
private utilityService: UtilityService;
|
private utilityService: UtilityService;
|
||||||
@@ -282,6 +252,12 @@ export class ApPersonService implements OnModuleInit {
|
|||||||
|
|
||||||
const bday = person['vcard:bday']?.match(/^\d{4}-\d{2}-\d{2}/);
|
const bday = person['vcard:bday']?.match(/^\d{4}-\d{2}-\d{2}/);
|
||||||
|
|
||||||
|
const url = getOneApHrefNullable(person.url);
|
||||||
|
|
||||||
|
if (url && !url.startsWith('https://')) {
|
||||||
|
throw new Error('unexpected shcema of person url: ' + url);
|
||||||
|
}
|
||||||
|
|
||||||
// Create user
|
// Create user
|
||||||
let user: IRemoteUser;
|
let user: IRemoteUser;
|
||||||
try {
|
try {
|
||||||
@@ -313,7 +289,7 @@ export class ApPersonService implements OnModuleInit {
|
|||||||
await transactionalEntityManager.save(new UserProfile({
|
await transactionalEntityManager.save(new UserProfile({
|
||||||
userId: user.id,
|
userId: user.id,
|
||||||
description: person.summary ? this.apMfmService.htmlToMfm(truncate(person.summary, summaryLength), person.tag) : null,
|
description: person.summary ? this.apMfmService.htmlToMfm(truncate(person.summary, summaryLength), person.tag) : null,
|
||||||
url: getOneApHrefNullable(person.url),
|
url: url,
|
||||||
fields,
|
fields,
|
||||||
birthday: bday ? bday[0] : null,
|
birthday: bday ? bday[0] : null,
|
||||||
location: person['vcard:Address'] ?? null,
|
location: person['vcard:Address'] ?? null,
|
||||||
@@ -455,6 +431,12 @@ export class ApPersonService implements OnModuleInit {
|
|||||||
|
|
||||||
const bday = person['vcard:bday']?.match(/^\d{4}-\d{2}-\d{2}/);
|
const bday = person['vcard:bday']?.match(/^\d{4}-\d{2}-\d{2}/);
|
||||||
|
|
||||||
|
const url = getOneApHrefNullable(person.url);
|
||||||
|
|
||||||
|
if (url && !url.startsWith('https://')) {
|
||||||
|
throw new Error('unexpected shcema of person url: ' + url);
|
||||||
|
}
|
||||||
|
|
||||||
const updates = {
|
const updates = {
|
||||||
lastFetchedAt: new Date(),
|
lastFetchedAt: new Date(),
|
||||||
inbox: person.inbox,
|
inbox: person.inbox,
|
||||||
@@ -489,7 +471,7 @@ export class ApPersonService implements OnModuleInit {
|
|||||||
}
|
}
|
||||||
|
|
||||||
await this.userProfilesRepository.update({ userId: exist.id }, {
|
await this.userProfilesRepository.update({ userId: exist.id }, {
|
||||||
url: getOneApHrefNullable(person.url),
|
url: url,
|
||||||
fields,
|
fields,
|
||||||
description: person.summary ? this.apMfmService.htmlToMfm(truncate(person.summary, summaryLength), person.tag) : null,
|
description: person.summary ? this.apMfmService.htmlToMfm(truncate(person.summary, summaryLength), person.tag) : null,
|
||||||
birthday: bday ? bday[0] : null,
|
birthday: bday ? bday[0] : null,
|
||||||
@@ -540,22 +522,16 @@ export class ApPersonService implements OnModuleInit {
|
|||||||
name: string,
|
name: string,
|
||||||
value: string
|
value: string
|
||||||
}[] = [];
|
}[] = [];
|
||||||
const services: { [x: string]: any } = {};
|
|
||||||
|
|
||||||
if (Array.isArray(attachments)) {
|
if (Array.isArray(attachments)) {
|
||||||
for (const attachment of attachments.filter(isPropertyValue)) {
|
for (const attachment of attachments.filter(isPropertyValue)) {
|
||||||
if (isPropertyValue(attachment.identifier)) {
|
fields.push({
|
||||||
addService(services, attachment.identifier);
|
name: attachment.name,
|
||||||
} else {
|
value: this.mfmService.fromHtml(attachment.value),
|
||||||
fields.push({
|
});
|
||||||
name: attachment.name,
|
|
||||||
value: this.mfmService.fromHtml(attachment.value),
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return { fields, services };
|
return { fields };
|
||||||
}
|
}
|
||||||
|
|
||||||
@bindThis
|
@bindThis
|
||||||
@@ -566,22 +542,22 @@ export class ApPersonService implements OnModuleInit {
|
|||||||
|
|
||||||
this.logger.info(`Updating the featured: ${user.uri}`);
|
this.logger.info(`Updating the featured: ${user.uri}`);
|
||||||
|
|
||||||
if (resolver == null) resolver = this.apResolverService.createResolver();
|
const _resolver = resolver ?? this.apResolverService.createResolver();
|
||||||
|
|
||||||
// Resolve to (Ordered)Collection Object
|
// Resolve to (Ordered)Collection Object
|
||||||
const collection = await resolver.resolveCollection(user.featured);
|
const collection = await _resolver.resolveCollection(user.featured);
|
||||||
if (!isCollectionOrOrderedCollection(collection)) throw new Error('Object is not Collection or OrderedCollection');
|
if (!isCollectionOrOrderedCollection(collection)) throw new Error('Object is not Collection or OrderedCollection');
|
||||||
|
|
||||||
// Resolve to Object(may be Note) arrays
|
// Resolve to Object(may be Note) arrays
|
||||||
const unresolvedItems = isCollection(collection) ? collection.items : collection.orderedItems;
|
const unresolvedItems = isCollection(collection) ? collection.items : collection.orderedItems;
|
||||||
const items = await Promise.all(toArray(unresolvedItems).map(x => resolver.resolve(x)));
|
const items = await Promise.all(toArray(unresolvedItems).map(x => _resolver.resolve(x)));
|
||||||
|
|
||||||
// Resolve and regist Notes
|
// Resolve and regist Notes
|
||||||
const limit = promiseLimit<Note | null>(2);
|
const limit = promiseLimit<Note | null>(2);
|
||||||
const featuredNotes = await Promise.all(items
|
const featuredNotes = await Promise.all(items
|
||||||
.filter(item => getApType(item) === 'Note') // TODO: Noteでなくてもいいかも
|
.filter(item => getApType(item) === 'Note') // TODO: Noteでなくてもいいかも
|
||||||
.slice(0, 5)
|
.slice(0, 5)
|
||||||
.map(item => limit(() => this.apNoteService.resolveNote(item, resolver))));
|
.map(item => limit(() => this.apNoteService.resolveNote(item, _resolver))));
|
||||||
|
|
||||||
await this.db.transaction(async transactionalEntityManager => {
|
await this.db.transaction(async transactionalEntityManager => {
|
||||||
await transactionalEntityManager.delete(UserNotePining, { userId: user.id });
|
await transactionalEntityManager.delete(UserNotePining, { userId: user.id });
|
||||||
|
@@ -10,7 +10,6 @@ import PerUserNotesChart from './charts/per-user-notes.js';
|
|||||||
import PerUserPvChart from './charts/per-user-pv.js';
|
import PerUserPvChart from './charts/per-user-pv.js';
|
||||||
import DriveChart from './charts/drive.js';
|
import DriveChart from './charts/drive.js';
|
||||||
import PerUserReactionsChart from './charts/per-user-reactions.js';
|
import PerUserReactionsChart from './charts/per-user-reactions.js';
|
||||||
import HashtagChart from './charts/hashtag.js';
|
|
||||||
import PerUserFollowingChart from './charts/per-user-following.js';
|
import PerUserFollowingChart from './charts/per-user-following.js';
|
||||||
import PerUserDriveChart from './charts/per-user-drive.js';
|
import PerUserDriveChart from './charts/per-user-drive.js';
|
||||||
import ApRequestChart from './charts/ap-request.js';
|
import ApRequestChart from './charts/ap-request.js';
|
||||||
@@ -31,7 +30,6 @@ export class ChartManagementService implements OnApplicationShutdown {
|
|||||||
private perUserPvChart: PerUserPvChart,
|
private perUserPvChart: PerUserPvChart,
|
||||||
private driveChart: DriveChart,
|
private driveChart: DriveChart,
|
||||||
private perUserReactionsChart: PerUserReactionsChart,
|
private perUserReactionsChart: PerUserReactionsChart,
|
||||||
private hashtagChart: HashtagChart,
|
|
||||||
private perUserFollowingChart: PerUserFollowingChart,
|
private perUserFollowingChart: PerUserFollowingChart,
|
||||||
private perUserDriveChart: PerUserDriveChart,
|
private perUserDriveChart: PerUserDriveChart,
|
||||||
private apRequestChart: ApRequestChart,
|
private apRequestChart: ApRequestChart,
|
||||||
@@ -46,7 +44,6 @@ export class ChartManagementService implements OnApplicationShutdown {
|
|||||||
this.perUserPvChart,
|
this.perUserPvChart,
|
||||||
this.driveChart,
|
this.driveChart,
|
||||||
this.perUserReactionsChart,
|
this.perUserReactionsChart,
|
||||||
this.hashtagChart,
|
|
||||||
this.perUserFollowingChart,
|
this.perUserFollowingChart,
|
||||||
this.perUserDriveChart,
|
this.perUserDriveChart,
|
||||||
this.apRequestChart,
|
this.apRequestChart,
|
||||||
|
@@ -1,10 +0,0 @@
|
|||||||
import Chart from '../../core.js';
|
|
||||||
|
|
||||||
export const name = 'hashtag';
|
|
||||||
|
|
||||||
export const schema = {
|
|
||||||
'local.users': { uniqueIncrement: true },
|
|
||||||
'remote.users': { uniqueIncrement: true },
|
|
||||||
} as const;
|
|
||||||
|
|
||||||
export const entity = Chart.schemaToEntity(name, schema, true);
|
|
@@ -1,45 +0,0 @@
|
|||||||
import { Injectable, Inject } from '@nestjs/common';
|
|
||||||
import { DataSource } from 'typeorm';
|
|
||||||
import type { User } from '@/models/entities/User.js';
|
|
||||||
import { AppLockService } from '@/core/AppLockService.js';
|
|
||||||
import { DI } from '@/di-symbols.js';
|
|
||||||
import { UserEntityService } from '@/core/entities/UserEntityService.js';
|
|
||||||
import { bindThis } from '@/decorators.js';
|
|
||||||
import Chart from '../core.js';
|
|
||||||
import { ChartLoggerService } from '../ChartLoggerService.js';
|
|
||||||
import { name, schema } from './entities/hashtag.js';
|
|
||||||
import type { KVs } from '../core.js';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* ハッシュタグに関するチャート
|
|
||||||
*/
|
|
||||||
// eslint-disable-next-line import/no-default-export
|
|
||||||
@Injectable()
|
|
||||||
export default class HashtagChart extends Chart<typeof schema> {
|
|
||||||
constructor(
|
|
||||||
@Inject(DI.db)
|
|
||||||
private db: DataSource,
|
|
||||||
|
|
||||||
private appLockService: AppLockService,
|
|
||||||
private userEntityService: UserEntityService,
|
|
||||||
private chartLoggerService: ChartLoggerService,
|
|
||||||
) {
|
|
||||||
super(db, (k) => appLockService.getChartInsertLock(k), chartLoggerService.logger, name, schema, true);
|
|
||||||
}
|
|
||||||
|
|
||||||
protected async tickMajor(): Promise<Partial<KVs<typeof schema>>> {
|
|
||||||
return {};
|
|
||||||
}
|
|
||||||
|
|
||||||
protected async tickMinor(): Promise<Partial<KVs<typeof schema>>> {
|
|
||||||
return {};
|
|
||||||
}
|
|
||||||
|
|
||||||
@bindThis
|
|
||||||
public async update(hashtag: string, user: { id: User['id'], host: User['host'] }): Promise<void> {
|
|
||||||
await this.commit({
|
|
||||||
'local.users': this.userEntityService.isLocalUser(user) ? [user.id] : [],
|
|
||||||
'remote.users': this.userEntityService.isLocalUser(user) ? [] : [user.id],
|
|
||||||
}, hashtag);
|
|
||||||
}
|
|
||||||
}
|
|
@@ -11,9 +11,9 @@ import type Logger from '@/logger.js';
|
|||||||
import { bindThis } from '@/decorators.js';
|
import { bindThis } from '@/decorators.js';
|
||||||
import type { Repository, DataSource } from 'typeorm';
|
import type { Repository, DataSource } from 'typeorm';
|
||||||
|
|
||||||
const columnPrefix = '___' as const;
|
const COLUMN_PREFIX = '___' as const;
|
||||||
const uniqueTempColumnPrefix = 'unique_temp___' as const;
|
const UNIQUE_TEMP_COLUMN_PREFIX = 'unique_temp___' as const;
|
||||||
const columnDot = '_' as const;
|
const COLUMN_DELIMITER = '_' as const;
|
||||||
|
|
||||||
type Schema = Record<string, {
|
type Schema = Record<string, {
|
||||||
uniqueIncrement?: boolean;
|
uniqueIncrement?: boolean;
|
||||||
@@ -26,14 +26,14 @@ type Schema = Record<string, {
|
|||||||
accumulate?: boolean;
|
accumulate?: boolean;
|
||||||
}>;
|
}>;
|
||||||
|
|
||||||
type KeyToColumnName<T extends string> = T extends `${infer R1}.${infer R2}` ? `${R1}${typeof columnDot}${KeyToColumnName<R2>}` : T;
|
type KeyToColumnName<T extends string> = T extends `${infer R1}.${infer R2}` ? `${R1}${typeof COLUMN_DELIMITER}${KeyToColumnName<R2>}` : T;
|
||||||
|
|
||||||
type Columns<S extends Schema> = {
|
type Columns<S extends Schema> = {
|
||||||
[K in keyof S as `${typeof columnPrefix}${KeyToColumnName<string & K>}`]: number;
|
[K in keyof S as `${typeof COLUMN_PREFIX}${KeyToColumnName<string & K>}`]: number;
|
||||||
};
|
};
|
||||||
|
|
||||||
type TempColumnsForUnique<S extends Schema> = {
|
type TempColumnsForUnique<S extends Schema> = {
|
||||||
[K in keyof S as `${typeof uniqueTempColumnPrefix}${KeyToColumnName<string & K>}`]: S[K]['uniqueIncrement'] extends true ? string[] : never;
|
[K in keyof S as `${typeof UNIQUE_TEMP_COLUMN_PREFIX}${KeyToColumnName<string & K>}`]: S[K]['uniqueIncrement'] extends true ? string[] : never;
|
||||||
};
|
};
|
||||||
|
|
||||||
type RawRecord<S extends Schema> = {
|
type RawRecord<S extends Schema> = {
|
||||||
@@ -138,20 +138,20 @@ export default abstract class Chart<T extends Schema> {
|
|||||||
private static convertSchemaToColumnDefinitions(schema: Schema): Record<string, { type: string; array?: boolean; default?: any; }> {
|
private static convertSchemaToColumnDefinitions(schema: Schema): Record<string, { type: string; array?: boolean; default?: any; }> {
|
||||||
const columns = {} as Record<string, { type: string; array?: boolean; default?: any; }>;
|
const columns = {} as Record<string, { type: string; array?: boolean; default?: any; }>;
|
||||||
for (const [k, v] of Object.entries(schema)) {
|
for (const [k, v] of Object.entries(schema)) {
|
||||||
const name = k.replaceAll('.', columnDot);
|
const name = k.replaceAll('.', COLUMN_DELIMITER);
|
||||||
const type = v.range === 'big' ? 'bigint' : v.range === 'small' ? 'smallint' : 'integer';
|
const type = v.range === 'big' ? 'bigint' : v.range === 'small' ? 'smallint' : 'integer';
|
||||||
if (v.uniqueIncrement) {
|
if (v.uniqueIncrement) {
|
||||||
columns[uniqueTempColumnPrefix + name] = {
|
columns[UNIQUE_TEMP_COLUMN_PREFIX + name] = {
|
||||||
type: 'varchar',
|
type: 'varchar',
|
||||||
array: true,
|
array: true,
|
||||||
default: '{}',
|
default: '{}',
|
||||||
};
|
};
|
||||||
columns[columnPrefix + name] = {
|
columns[COLUMN_PREFIX + name] = {
|
||||||
type,
|
type,
|
||||||
default: 0,
|
default: 0,
|
||||||
};
|
};
|
||||||
} else {
|
} else {
|
||||||
columns[columnPrefix + name] = {
|
columns[COLUMN_PREFIX + name] = {
|
||||||
type,
|
type,
|
||||||
default: 0,
|
default: 0,
|
||||||
};
|
};
|
||||||
@@ -253,8 +253,8 @@ export default abstract class Chart<T extends Schema> {
|
|||||||
@bindThis
|
@bindThis
|
||||||
private convertRawRecord(x: RawRecord<T>): KVs<T> {
|
private convertRawRecord(x: RawRecord<T>): KVs<T> {
|
||||||
const kvs = {} as Record<string, number>;
|
const kvs = {} as Record<string, number>;
|
||||||
for (const k of Object.keys(x).filter((k) => k.startsWith(columnPrefix)) as (keyof Columns<T>)[]) {
|
for (const k of Object.keys(x).filter((k) => k.startsWith(COLUMN_PREFIX)) as (keyof Columns<T>)[]) {
|
||||||
kvs[(k as string).substr(columnPrefix.length).split(columnDot).join('.')] = x[k] as unknown as number;
|
kvs[(k as string).substr(COLUMN_PREFIX.length).split(COLUMN_DELIMITER).join('.')] = x[k] as unknown as number;
|
||||||
}
|
}
|
||||||
return kvs as KVs<T>;
|
return kvs as KVs<T>;
|
||||||
}
|
}
|
||||||
@@ -357,8 +357,8 @@ export default abstract class Chart<T extends Schema> {
|
|||||||
|
|
||||||
const columns = {} as Record<string, number | unknown[]>;
|
const columns = {} as Record<string, number | unknown[]>;
|
||||||
for (const [k, v] of Object.entries(data)) {
|
for (const [k, v] of Object.entries(data)) {
|
||||||
const name = k.replaceAll('.', columnDot);
|
const name = k.replaceAll('.', COLUMN_DELIMITER);
|
||||||
columns[columnPrefix + name] = v;
|
columns[COLUMN_PREFIX + name] = v;
|
||||||
}
|
}
|
||||||
|
|
||||||
// 新規ログ挿入
|
// 新規ログ挿入
|
||||||
@@ -419,13 +419,13 @@ export default abstract class Chart<T extends Schema> {
|
|||||||
const queryForDay: Record<keyof RawRecord<T>, number | (() => string)> = {} as any;
|
const queryForDay: Record<keyof RawRecord<T>, number | (() => string)> = {} as any;
|
||||||
for (const [k, v] of Object.entries(finalDiffs)) {
|
for (const [k, v] of Object.entries(finalDiffs)) {
|
||||||
if (typeof v === 'number') {
|
if (typeof v === 'number') {
|
||||||
const name = columnPrefix + k.replaceAll('.', columnDot) as string & keyof Columns<T>;
|
const name = COLUMN_PREFIX + k.replaceAll('.', COLUMN_DELIMITER) as string & keyof Columns<T>;
|
||||||
if (v > 0) queryForHour[name] = () => `"${name}" + ${v}`;
|
if (v > 0) queryForHour[name] = () => `"${name}" + ${v}`;
|
||||||
if (v < 0) queryForHour[name] = () => `"${name}" - ${Math.abs(v)}`;
|
if (v < 0) queryForHour[name] = () => `"${name}" - ${Math.abs(v)}`;
|
||||||
if (v > 0) queryForDay[name] = () => `"${name}" + ${v}`;
|
if (v > 0) queryForDay[name] = () => `"${name}" + ${v}`;
|
||||||
if (v < 0) queryForDay[name] = () => `"${name}" - ${Math.abs(v)}`;
|
if (v < 0) queryForDay[name] = () => `"${name}" - ${Math.abs(v)}`;
|
||||||
} else if (Array.isArray(v) && v.length > 0) { // ユニークインクリメント
|
} else if (Array.isArray(v) && v.length > 0) { // ユニークインクリメント
|
||||||
const tempColumnName = uniqueTempColumnPrefix + k.replaceAll('.', columnDot) as string & keyof TempColumnsForUnique<T>;
|
const tempColumnName = UNIQUE_TEMP_COLUMN_PREFIX + k.replaceAll('.', COLUMN_DELIMITER) as string & keyof TempColumnsForUnique<T>;
|
||||||
// TODO: item をSQLエスケープ
|
// TODO: item をSQLエスケープ
|
||||||
const itemsForHour = v.filter(item => !(logHour[tempColumnName] as unknown as string[]).includes(item)).map(item => `"${item}"`);
|
const itemsForHour = v.filter(item => !(logHour[tempColumnName] as unknown as string[]).includes(item)).map(item => `"${item}"`);
|
||||||
const itemsForDay = v.filter(item => !(logDay[tempColumnName] as unknown as string[]).includes(item)).map(item => `"${item}"`);
|
const itemsForDay = v.filter(item => !(logDay[tempColumnName] as unknown as string[]).includes(item)).map(item => `"${item}"`);
|
||||||
@@ -437,8 +437,8 @@ export default abstract class Chart<T extends Schema> {
|
|||||||
// bake unique count
|
// bake unique count
|
||||||
for (const [k, v] of Object.entries(finalDiffs)) {
|
for (const [k, v] of Object.entries(finalDiffs)) {
|
||||||
if (this.schema[k].uniqueIncrement) {
|
if (this.schema[k].uniqueIncrement) {
|
||||||
const name = columnPrefix + k.replaceAll('.', columnDot) as keyof Columns<T>;
|
const name = COLUMN_PREFIX + k.replaceAll('.', COLUMN_DELIMITER) as keyof Columns<T>;
|
||||||
const tempColumnName = uniqueTempColumnPrefix + k.replaceAll('.', columnDot) as keyof TempColumnsForUnique<T>;
|
const tempColumnName = UNIQUE_TEMP_COLUMN_PREFIX + k.replaceAll('.', COLUMN_DELIMITER) as keyof TempColumnsForUnique<T>;
|
||||||
queryForHour[name] = new Set([...(v as string[]), ...(logHour[tempColumnName] as unknown as string[])]).size;
|
queryForHour[name] = new Set([...(v as string[]), ...(logHour[tempColumnName] as unknown as string[])]).size;
|
||||||
queryForDay[name] = new Set([...(v as string[]), ...(logDay[tempColumnName] as unknown as string[])]).size;
|
queryForDay[name] = new Set([...(v as string[]), ...(logDay[tempColumnName] as unknown as string[])]).size;
|
||||||
}
|
}
|
||||||
@@ -449,15 +449,15 @@ export default abstract class Chart<T extends Schema> {
|
|||||||
for (const [k, v] of Object.entries(this.schema)) {
|
for (const [k, v] of Object.entries(this.schema)) {
|
||||||
const intersection = v.intersection;
|
const intersection = v.intersection;
|
||||||
if (intersection) {
|
if (intersection) {
|
||||||
const name = columnPrefix + k.replaceAll('.', columnDot) as keyof Columns<T>;
|
const name = COLUMN_PREFIX + k.replaceAll('.', COLUMN_DELIMITER) as keyof Columns<T>;
|
||||||
const firstKey = intersection[0];
|
const firstKey = intersection[0];
|
||||||
const firstTempColumnName = uniqueTempColumnPrefix + firstKey.replaceAll('.', columnDot) as keyof TempColumnsForUnique<T>;
|
const firstTempColumnName = UNIQUE_TEMP_COLUMN_PREFIX + firstKey.replaceAll('.', COLUMN_DELIMITER) as keyof TempColumnsForUnique<T>;
|
||||||
const firstValues = finalDiffs[firstKey] as string[] | undefined;
|
const firstValues = finalDiffs[firstKey] as string[] | undefined;
|
||||||
const currentValuesForHour = new Set([...(firstValues ?? []), ...(logHour[firstTempColumnName] as unknown as string[])]);
|
const currentValuesForHour = new Set([...(firstValues ?? []), ...(logHour[firstTempColumnName] as unknown as string[])]);
|
||||||
const currentValuesForDay = new Set([...(firstValues ?? []), ...(logDay[firstTempColumnName] as unknown as string[])]);
|
const currentValuesForDay = new Set([...(firstValues ?? []), ...(logDay[firstTempColumnName] as unknown as string[])]);
|
||||||
for (let i = 1; i < intersection.length; i++) {
|
for (let i = 1; i < intersection.length; i++) {
|
||||||
const targetKey = intersection[i];
|
const targetKey = intersection[i];
|
||||||
const targetTempColumnName = uniqueTempColumnPrefix + targetKey.replaceAll('.', columnDot) as keyof TempColumnsForUnique<T>;
|
const targetTempColumnName = UNIQUE_TEMP_COLUMN_PREFIX + targetKey.replaceAll('.', COLUMN_DELIMITER) as keyof TempColumnsForUnique<T>;
|
||||||
const targetValues = finalDiffs[targetKey] as string[] | undefined;
|
const targetValues = finalDiffs[targetKey] as string[] | undefined;
|
||||||
const targetValuesForHour = new Set([...(targetValues ?? []), ...(logHour[targetTempColumnName] as unknown as string[])]);
|
const targetValuesForHour = new Set([...(targetValues ?? []), ...(logHour[targetTempColumnName] as unknown as string[])]);
|
||||||
const targetValuesForDay = new Set([...(targetValues ?? []), ...(logDay[targetTempColumnName] as unknown as string[])]);
|
const targetValuesForDay = new Set([...(targetValues ?? []), ...(logDay[targetTempColumnName] as unknown as string[])]);
|
||||||
@@ -510,7 +510,7 @@ export default abstract class Chart<T extends Schema> {
|
|||||||
|
|
||||||
const columns = {} as Record<keyof Columns<T>, number>;
|
const columns = {} as Record<keyof Columns<T>, number>;
|
||||||
for (const [k, v] of Object.entries(data) as ([keyof typeof data, number])[]) {
|
for (const [k, v] of Object.entries(data) as ([keyof typeof data, number])[]) {
|
||||||
const name = columnPrefix + (k as string).replaceAll('.', columnDot) as keyof Columns<T>;
|
const name = COLUMN_PREFIX + (k as string).replaceAll('.', COLUMN_DELIMITER) as keyof Columns<T>;
|
||||||
columns[name] = v;
|
columns[name] = v;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -556,7 +556,7 @@ export default abstract class Chart<T extends Schema> {
|
|||||||
const columns = {} as Record<keyof TempColumnsForUnique<T>, []>;
|
const columns = {} as Record<keyof TempColumnsForUnique<T>, []>;
|
||||||
for (const [k, v] of Object.entries(this.schema)) {
|
for (const [k, v] of Object.entries(this.schema)) {
|
||||||
if (v.uniqueIncrement) {
|
if (v.uniqueIncrement) {
|
||||||
const name = uniqueTempColumnPrefix + k.replaceAll('.', columnDot) as keyof TempColumnsForUnique<T>;
|
const name = UNIQUE_TEMP_COLUMN_PREFIX + k.replaceAll('.', COLUMN_DELIMITER) as keyof TempColumnsForUnique<T>;
|
||||||
columns[name] = [];
|
columns[name] = [];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -7,7 +7,6 @@ import { entity as PerUserNotesChart } from './charts/entities/per-user-notes.js
|
|||||||
import { entity as PerUserPvChart } from './charts/entities/per-user-pv.js';
|
import { entity as PerUserPvChart } from './charts/entities/per-user-pv.js';
|
||||||
import { entity as DriveChart } from './charts/entities/drive.js';
|
import { entity as DriveChart } from './charts/entities/drive.js';
|
||||||
import { entity as PerUserReactionsChart } from './charts/entities/per-user-reactions.js';
|
import { entity as PerUserReactionsChart } from './charts/entities/per-user-reactions.js';
|
||||||
import { entity as HashtagChart } from './charts/entities/hashtag.js';
|
|
||||||
import { entity as PerUserFollowingChart } from './charts/entities/per-user-following.js';
|
import { entity as PerUserFollowingChart } from './charts/entities/per-user-following.js';
|
||||||
import { entity as PerUserDriveChart } from './charts/entities/per-user-drive.js';
|
import { entity as PerUserDriveChart } from './charts/entities/per-user-drive.js';
|
||||||
import { entity as ApRequestChart } from './charts/entities/ap-request.js';
|
import { entity as ApRequestChart } from './charts/entities/ap-request.js';
|
||||||
@@ -27,7 +26,6 @@ export const entities = [
|
|||||||
PerUserPvChart.hour, PerUserPvChart.day,
|
PerUserPvChart.hour, PerUserPvChart.day,
|
||||||
DriveChart.hour, DriveChart.day,
|
DriveChart.hour, DriveChart.day,
|
||||||
PerUserReactionsChart.hour, PerUserReactionsChart.day,
|
PerUserReactionsChart.hour, PerUserReactionsChart.day,
|
||||||
HashtagChart.hour, HashtagChart.day,
|
|
||||||
PerUserFollowingChart.hour, PerUserFollowingChart.day,
|
PerUserFollowingChart.hour, PerUserFollowingChart.day,
|
||||||
PerUserDriveChart.hour, PerUserDriveChart.day,
|
PerUserDriveChart.hour, PerUserDriveChart.day,
|
||||||
ApRequestChart.hour, ApRequestChart.day,
|
ApRequestChart.hour, ApRequestChart.day,
|
||||||
|
@@ -54,7 +54,7 @@ export class ChannelEntityService {
|
|||||||
name: channel.name,
|
name: channel.name,
|
||||||
description: channel.description,
|
description: channel.description,
|
||||||
userId: channel.userId,
|
userId: channel.userId,
|
||||||
bannerUrl: banner ? this.driveFileEntityService.getPublicUrl(banner, false) : null,
|
bannerUrl: banner ? this.driveFileEntityService.getPublicUrl(banner) : null,
|
||||||
usersCount: channel.usersCount,
|
usersCount: channel.usersCount,
|
||||||
notesCount: channel.notesCount,
|
notesCount: channel.notesCount,
|
||||||
|
|
||||||
|
@@ -20,6 +20,7 @@ type PackOptions = {
|
|||||||
withUser?: boolean,
|
withUser?: boolean,
|
||||||
};
|
};
|
||||||
import { bindThis } from '@/decorators.js';
|
import { bindThis } from '@/decorators.js';
|
||||||
|
import { isMimeImage } from '@/misc/is-mime-image.js';
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class DriveFileEntityService {
|
export class DriveFileEntityService {
|
||||||
@@ -71,27 +72,42 @@ export class DriveFileEntityService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@bindThis
|
@bindThis
|
||||||
public getPublicUrl(file: DriveFile, thumbnail = false): string | null {
|
public getPublicUrl(file: DriveFile, mode? : 'static' | 'avatar'): string | null { // static = thumbnail
|
||||||
|
const proxiedUrl = (url: string) => appendQuery(
|
||||||
|
`${this.config.mediaProxy}/${mode ?? 'image'}.webp`,
|
||||||
|
query({
|
||||||
|
url,
|
||||||
|
...(mode ? { [mode]: '1' } : {}),
|
||||||
|
})
|
||||||
|
);
|
||||||
|
|
||||||
// リモートかつメディアプロキシ
|
// リモートかつメディアプロキシ
|
||||||
if (file.uri != null && file.userHost != null && this.config.mediaProxy != null) {
|
if (file.uri != null && file.userHost != null && this.config.externalMediaProxyEnabled) {
|
||||||
return appendQuery(this.config.mediaProxy, query({
|
if (!(mode === 'static' && file.type.startsWith('video'))) {
|
||||||
url: file.uri,
|
return proxiedUrl(file.uri);
|
||||||
thumbnail: thumbnail ? '1' : undefined,
|
}
|
||||||
}));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// リモートかつ期限切れはローカルプロキシを試みる
|
// リモートかつ期限切れはローカルプロキシを試みる
|
||||||
if (file.uri != null && file.isLink && this.config.proxyRemoteFiles) {
|
if (file.uri != null && file.isLink && this.config.proxyRemoteFiles) {
|
||||||
const key = thumbnail ? file.thumbnailAccessKey : file.webpublicAccessKey;
|
const key = mode === 'static' ? file.thumbnailAccessKey : file.webpublicAccessKey;
|
||||||
|
|
||||||
if (key && !key.match('/')) { // 古いものはここにオブジェクトストレージキーが入ってるので除外
|
if (key && !key.match('/')) { // 古いものはここにオブジェクトストレージキーが入ってるので除外
|
||||||
return `${this.config.url}/files/${key}`;
|
const url = `${this.config.url}/files/${key}`;
|
||||||
|
if (mode === 'avatar') return proxiedUrl(file.uri);
|
||||||
|
return url;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const isImage = file.type && ['image/png', 'image/apng', 'image/gif', 'image/jpeg', 'image/webp', 'image/avif', 'image/svg+xml'].includes(file.type);
|
const url = file.webpublicUrl ?? file.url;
|
||||||
|
|
||||||
return thumbnail ? (file.thumbnailUrl ?? (isImage ? (file.webpublicUrl ?? file.url) : null)) : (file.webpublicUrl ?? file.url);
|
if (mode === 'static') {
|
||||||
|
return file.thumbnailUrl ?? (isMimeImage(file.type, 'sharp-convertible-image') ? proxiedUrl(url) : null);
|
||||||
|
}
|
||||||
|
if (mode === 'avatar') {
|
||||||
|
return proxiedUrl(url);
|
||||||
|
}
|
||||||
|
return url;
|
||||||
}
|
}
|
||||||
|
|
||||||
@bindThis
|
@bindThis
|
||||||
@@ -166,8 +182,8 @@ export class DriveFileEntityService {
|
|||||||
isSensitive: file.isSensitive,
|
isSensitive: file.isSensitive,
|
||||||
blurhash: file.blurhash,
|
blurhash: file.blurhash,
|
||||||
properties: opts.self ? file.properties : this.getPublicProperties(file),
|
properties: opts.self ? file.properties : this.getPublicProperties(file),
|
||||||
url: opts.self ? file.url : this.getPublicUrl(file, false),
|
url: opts.self ? file.url : this.getPublicUrl(file),
|
||||||
thumbnailUrl: this.getPublicUrl(file, true),
|
thumbnailUrl: this.getPublicUrl(file, 'static'),
|
||||||
comment: file.comment,
|
comment: file.comment,
|
||||||
folderId: file.folderId,
|
folderId: file.folderId,
|
||||||
folder: opts.detail && file.folderId ? this.driveFolderEntityService.pack(file.folderId, {
|
folder: opts.detail && file.folderId ? this.driveFolderEntityService.pack(file.folderId, {
|
||||||
@@ -201,8 +217,8 @@ export class DriveFileEntityService {
|
|||||||
isSensitive: file.isSensitive,
|
isSensitive: file.isSensitive,
|
||||||
blurhash: file.blurhash,
|
blurhash: file.blurhash,
|
||||||
properties: opts.self ? file.properties : this.getPublicProperties(file),
|
properties: opts.self ? file.properties : this.getPublicProperties(file),
|
||||||
url: opts.self ? file.url : this.getPublicUrl(file, false),
|
url: opts.self ? file.url : this.getPublicUrl(file),
|
||||||
thumbnailUrl: this.getPublicUrl(file, true),
|
thumbnailUrl: this.getPublicUrl(file, 'static'),
|
||||||
comment: file.comment,
|
comment: file.comment,
|
||||||
folderId: file.folderId,
|
folderId: file.folderId,
|
||||||
folder: opts.detail && file.folderId ? this.driveFolderEntityService.pack(file.folderId, {
|
folder: opts.detail && file.folderId ? this.driveFolderEntityService.pack(file.folderId, {
|
||||||
|
@@ -22,8 +22,10 @@ export class EmojiEntityService {
|
|||||||
@bindThis
|
@bindThis
|
||||||
public async pack(
|
public async pack(
|
||||||
src: Emoji['id'] | Emoji,
|
src: Emoji['id'] | Emoji,
|
||||||
opts: { omitHost?: boolean; omitId?: boolean; withUrl?: boolean; } = {},
|
opts: { omitHost?: boolean; omitId?: boolean; withUrl?: boolean; } = { omitHost: true, omitId: true, withUrl: true },
|
||||||
): Promise<Packed<'Emoji'>> {
|
): Promise<Packed<'Emoji'>> {
|
||||||
|
opts = { omitHost: true, omitId: true, withUrl: true, ...opts }
|
||||||
|
|
||||||
const emoji = typeof src === 'object' ? src : await this.emojisRepository.findOneByOrFail({ id: src });
|
const emoji = typeof src === 'object' ? src : await this.emojisRepository.findOneByOrFail({ id: src });
|
||||||
|
|
||||||
return {
|
return {
|
||||||
@@ -32,8 +34,8 @@ export class EmojiEntityService {
|
|||||||
name: emoji.name,
|
name: emoji.name,
|
||||||
category: emoji.category,
|
category: emoji.category,
|
||||||
host: opts.omitHost ? undefined : emoji.host,
|
host: opts.omitHost ? undefined : emoji.host,
|
||||||
// ?? emoji.originalUrl してるのは後方互換性のため
|
// || emoji.originalUrl してるのは後方互換性のため(publicUrlはstringなので??はだめ)
|
||||||
url: opts.withUrl ? (emoji.publicUrl ?? emoji.originalUrl) : undefined,
|
url: opts.withUrl ? (emoji.publicUrl || emoji.originalUrl) : undefined,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -282,7 +282,9 @@ export class NoteEntityService implements OnModuleInit {
|
|||||||
: await this.channelsRepository.findOneBy({ id: note.channelId })
|
: await this.channelsRepository.findOneBy({ id: note.channelId })
|
||||||
: null;
|
: null;
|
||||||
|
|
||||||
const reactionEmojiNames = Object.keys(note.reactions).filter(x => x.startsWith(':')).map(x => this.reactionService.decodeReaction(x).reaction).map(x => x.replace(/:/g, ''));
|
const reactionEmojiNames = Object.keys(note.reactions)
|
||||||
|
.filter(x => x.startsWith(':') && x.includes('@') && !x.includes('@.')) // リモートカスタム絵文字のみ
|
||||||
|
.map(x => this.reactionService.decodeReaction(x).reaction.replaceAll(':', ''));
|
||||||
|
|
||||||
const packed: Packed<'Note'> = await awaitAll({
|
const packed: Packed<'Note'> = await awaitAll({
|
||||||
id: note.id,
|
id: note.id,
|
||||||
@@ -299,6 +301,8 @@ export class NoteEntityService implements OnModuleInit {
|
|||||||
renoteCount: note.renoteCount,
|
renoteCount: note.renoteCount,
|
||||||
repliesCount: note.repliesCount,
|
repliesCount: note.repliesCount,
|
||||||
reactions: this.reactionService.convertLegacyReactions(note.reactions),
|
reactions: this.reactionService.convertLegacyReactions(note.reactions),
|
||||||
|
reactionEmojis: this.customEmojiService.populateEmojis(reactionEmojiNames, host),
|
||||||
|
emojis: host != null ? this.customEmojiService.populateEmojis(note.emojis, host) : undefined,
|
||||||
tags: note.tags.length > 0 ? note.tags : undefined,
|
tags: note.tags.length > 0 ? note.tags : undefined,
|
||||||
fileIds: note.fileIds,
|
fileIds: note.fileIds,
|
||||||
files: this.driveFileEntityService.packMany(note.fileIds),
|
files: this.driveFileEntityService.packMany(note.fileIds),
|
||||||
@@ -384,6 +388,8 @@ export class NoteEntityService implements OnModuleInit {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
await this.customEmojiService.prefetchEmojis(this.customEmojiService.aggregateNoteEmojis(notes));
|
||||||
|
|
||||||
return await Promise.all(notes.map(n => this.pack(n, me, {
|
return await Promise.all(notes.map(n => this.pack(n, me, {
|
||||||
...options,
|
...options,
|
||||||
_hint_: {
|
_hint_: {
|
||||||
|
@@ -2,7 +2,7 @@ import { Inject, Injectable } from '@nestjs/common';
|
|||||||
import { In } from 'typeorm';
|
import { In } from 'typeorm';
|
||||||
import { ModuleRef } from '@nestjs/core';
|
import { ModuleRef } from '@nestjs/core';
|
||||||
import { DI } from '@/di-symbols.js';
|
import { DI } from '@/di-symbols.js';
|
||||||
import type { AccessTokensRepository, NoteReactionsRepository, NotificationsRepository } from '@/models/index.js';
|
import type { AccessTokensRepository, NoteReactionsRepository, NotificationsRepository, User } from '@/models/index.js';
|
||||||
import { awaitAll } from '@/misc/prelude/await-all.js';
|
import { awaitAll } from '@/misc/prelude/await-all.js';
|
||||||
import type { Notification } from '@/models/entities/Notification.js';
|
import type { Notification } from '@/models/entities/Notification.js';
|
||||||
import type { NoteReaction } from '@/models/entities/NoteReaction.js';
|
import type { NoteReaction } from '@/models/entities/NoteReaction.js';
|
||||||
@@ -146,6 +146,8 @@ export class NotificationEntityService implements OnModuleInit {
|
|||||||
myReactionsMap.set(target, myReactions.find(reaction => reaction.noteId === target) ?? null);
|
myReactionsMap.set(target, myReactions.find(reaction => reaction.noteId === target) ?? null);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
await this.customEmojiService.prefetchEmojis(this.customEmojiService.aggregateNoteEmojis(notes));
|
||||||
|
|
||||||
return await Promise.all(notifications.map(x => this.pack(x, {
|
return await Promise.all(notifications.map(x => this.pack(x, {
|
||||||
_hintForEachNotes_: {
|
_hintForEachNotes_: {
|
||||||
myReactions: myReactionsMap,
|
myReactions: myReactionsMap,
|
||||||
|
@@ -56,11 +56,13 @@ export class RoleEntityService {
|
|||||||
name: role.name,
|
name: role.name,
|
||||||
description: role.description,
|
description: role.description,
|
||||||
color: role.color,
|
color: role.color,
|
||||||
|
iconUrl: role.iconUrl,
|
||||||
target: role.target,
|
target: role.target,
|
||||||
condFormula: role.condFormula,
|
condFormula: role.condFormula,
|
||||||
isPublic: role.isPublic,
|
isPublic: role.isPublic,
|
||||||
isAdministrator: role.isAdministrator,
|
isAdministrator: role.isAdministrator,
|
||||||
isModerator: role.isModerator,
|
isModerator: role.isModerator,
|
||||||
|
asBadge: role.asBadge,
|
||||||
canEditMembersByModerator: role.canEditMembersByModerator,
|
canEditMembersByModerator: role.canEditMembersByModerator,
|
||||||
policies: policies,
|
policies: policies,
|
||||||
usersCount: assigns.length,
|
usersCount: assigns.length,
|
||||||
|
@@ -314,10 +314,10 @@ export class UserEntityService implements OnModuleInit {
|
|||||||
@bindThis
|
@bindThis
|
||||||
public async getAvatarUrl(user: User): Promise<string> {
|
public async getAvatarUrl(user: User): Promise<string> {
|
||||||
if (user.avatar) {
|
if (user.avatar) {
|
||||||
return this.driveFileEntityService.getPublicUrl(user.avatar, true) ?? this.getIdenticonUrl(user.id);
|
return this.driveFileEntityService.getPublicUrl(user.avatar, 'avatar') ?? this.getIdenticonUrl(user.id);
|
||||||
} else if (user.avatarId) {
|
} else if (user.avatarId) {
|
||||||
const avatar = await this.driveFilesRepository.findOneByOrFail({ id: user.avatarId });
|
const avatar = await this.driveFilesRepository.findOneByOrFail({ id: user.avatarId });
|
||||||
return this.driveFileEntityService.getPublicUrl(avatar, true) ?? this.getIdenticonUrl(user.id);
|
return this.driveFileEntityService.getPublicUrl(avatar, 'avatar') ?? this.getIdenticonUrl(user.id);
|
||||||
} else {
|
} else {
|
||||||
return this.getIdenticonUrl(user.id);
|
return this.getIdenticonUrl(user.id);
|
||||||
}
|
}
|
||||||
@@ -326,7 +326,7 @@ export class UserEntityService implements OnModuleInit {
|
|||||||
@bindThis
|
@bindThis
|
||||||
public getAvatarUrlSync(user: User): string {
|
public getAvatarUrlSync(user: User): string {
|
||||||
if (user.avatar) {
|
if (user.avatar) {
|
||||||
return this.driveFileEntityService.getPublicUrl(user.avatar, true) ?? this.getIdenticonUrl(user.id);
|
return this.driveFileEntityService.getPublicUrl(user.avatar, 'avatar') ?? this.getIdenticonUrl(user.id);
|
||||||
} else {
|
} else {
|
||||||
return this.getIdenticonUrl(user.id);
|
return this.getIdenticonUrl(user.id);
|
||||||
}
|
}
|
||||||
@@ -413,7 +413,13 @@ export class UserEntityService implements OnModuleInit {
|
|||||||
faviconUrl: instance.faviconUrl,
|
faviconUrl: instance.faviconUrl,
|
||||||
themeColor: instance.themeColor,
|
themeColor: instance.themeColor,
|
||||||
} : undefined) : undefined,
|
} : undefined) : undefined,
|
||||||
|
emojis: this.customEmojiService.populateEmojis(user.emojis, user.host),
|
||||||
onlineStatus: this.getOnlineStatus(user),
|
onlineStatus: this.getOnlineStatus(user),
|
||||||
|
// パフォーマンス上の理由でローカルユーザーのみ
|
||||||
|
badgeRoles: user.host == null ? this.roleService.getUserBadgeRoles(user.id).then(rs => rs.map(r => ({
|
||||||
|
name: r.name,
|
||||||
|
iconUrl: r.iconUrl,
|
||||||
|
}))) : undefined,
|
||||||
|
|
||||||
...(opts.detail ? {
|
...(opts.detail ? {
|
||||||
url: profile!.url,
|
url: profile!.url,
|
||||||
@@ -421,7 +427,7 @@ export class UserEntityService implements OnModuleInit {
|
|||||||
createdAt: user.createdAt.toISOString(),
|
createdAt: user.createdAt.toISOString(),
|
||||||
updatedAt: user.updatedAt ? user.updatedAt.toISOString() : null,
|
updatedAt: user.updatedAt ? user.updatedAt.toISOString() : null,
|
||||||
lastFetchedAt: user.lastFetchedAt ? user.lastFetchedAt.toISOString() : null,
|
lastFetchedAt: user.lastFetchedAt ? user.lastFetchedAt.toISOString() : null,
|
||||||
bannerUrl: user.banner ? this.driveFileEntityService.getPublicUrl(user.banner, false) : null,
|
bannerUrl: user.banner ? this.driveFileEntityService.getPublicUrl(user.banner) : null,
|
||||||
bannerBlurhash: user.banner?.blurhash ?? null,
|
bannerBlurhash: user.banner?.blurhash ?? null,
|
||||||
isLocked: user.isLocked,
|
isLocked: user.isLocked,
|
||||||
isSilenced: this.roleService.getUserPolicies(user.id).then(r => !r.canPublicNote),
|
isSilenced: this.roleService.getUserPolicies(user.id).then(r => !r.canPublicNote),
|
||||||
@@ -453,6 +459,7 @@ export class UserEntityService implements OnModuleInit {
|
|||||||
id: role.id,
|
id: role.id,
|
||||||
name: role.name,
|
name: role.name,
|
||||||
color: role.color,
|
color: role.color,
|
||||||
|
iconUrl: role.iconUrl,
|
||||||
description: role.description,
|
description: role.description,
|
||||||
isModerator: role.isModerator,
|
isModerator: role.isModerator,
|
||||||
isAdministrator: role.isAdministrator,
|
isAdministrator: role.isAdministrator,
|
||||||
@@ -488,7 +495,6 @@ export class UserEntityService implements OnModuleInit {
|
|||||||
hasUnreadMessagingMessage: this.getHasUnreadMessagingMessage(user.id),
|
hasUnreadMessagingMessage: this.getHasUnreadMessagingMessage(user.id),
|
||||||
hasUnreadNotification: this.getHasUnreadNotification(user.id),
|
hasUnreadNotification: this.getHasUnreadNotification(user.id),
|
||||||
hasPendingReceivedFollowRequest: this.getHasPendingReceivedFollowRequest(user.id),
|
hasPendingReceivedFollowRequest: this.getHasPendingReceivedFollowRequest(user.id),
|
||||||
integrations: profile!.integrations,
|
|
||||||
mutedWords: profile!.mutedWords,
|
mutedWords: profile!.mutedWords,
|
||||||
mutedInstances: profile!.mutedInstances,
|
mutedInstances: profile!.mutedInstances,
|
||||||
mutingNotificationTypes: profile!.mutingNotificationTypes,
|
mutingNotificationTypes: profile!.mutingNotificationTypes,
|
||||||
@@ -496,10 +502,10 @@ export class UserEntityService implements OnModuleInit {
|
|||||||
showTimelineReplies: user.showTimelineReplies ?? falsy,
|
showTimelineReplies: user.showTimelineReplies ?? falsy,
|
||||||
achievements: profile!.achievements,
|
achievements: profile!.achievements,
|
||||||
loggedInDays: profile!.loggedInDates.length,
|
loggedInDays: profile!.loggedInDates.length,
|
||||||
|
policies: this.roleService.getUserPolicies(user.id),
|
||||||
} : {}),
|
} : {}),
|
||||||
|
|
||||||
...(opts.includeSecrets ? {
|
...(opts.includeSecrets ? {
|
||||||
policies: this.roleService.getUserPolicies(user.id),
|
|
||||||
email: profile!.email,
|
email: profile!.email,
|
||||||
emailVerified: profile!.emailVerified,
|
emailVerified: profile!.emailVerified,
|
||||||
securityKeysList: profile!.twoFactorEnabled
|
securityKeysList: profile!.twoFactorEnabled
|
||||||
|
@@ -5,7 +5,7 @@
|
|||||||
* The getter will return a .bind version of the function
|
* The getter will return a .bind version of the function
|
||||||
* and memoize the result against a symbol on the instance
|
* and memoize the result against a symbol on the instance
|
||||||
*/
|
*/
|
||||||
export function bindThis(target, key, descriptor) {
|
export function bindThis(target: any, key: string, descriptor: any) {
|
||||||
let fn = descriptor.value;
|
let fn = descriptor.value;
|
||||||
|
|
||||||
if (typeof fn !== 'function') {
|
if (typeof fn !== 'function') {
|
||||||
@@ -34,7 +34,7 @@ export function bindThis(target, key, descriptor) {
|
|||||||
});
|
});
|
||||||
return boundFn;
|
return boundFn;
|
||||||
},
|
},
|
||||||
set(value) {
|
set(value: any) {
|
||||||
fn = value;
|
fn = value;
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
@@ -17,15 +17,13 @@ export default class Logger {
|
|||||||
private context: Context;
|
private context: Context;
|
||||||
private parentLogger: Logger | null = null;
|
private parentLogger: Logger | null = null;
|
||||||
private store: boolean;
|
private store: boolean;
|
||||||
private syslogClient: any | null = null;
|
|
||||||
|
|
||||||
constructor(context: string, color?: KEYWORD, store = true, syslogClient = null) {
|
constructor(context: string, color?: KEYWORD, store = true) {
|
||||||
this.context = {
|
this.context = {
|
||||||
name: context,
|
name: context,
|
||||||
color: color,
|
color: color,
|
||||||
};
|
};
|
||||||
this.store = store;
|
this.store = store;
|
||||||
this.syslogClient = syslogClient;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@bindThis
|
@bindThis
|
||||||
@@ -47,7 +45,7 @@ export default class Logger {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const time = dateFormat(new Date(), 'HH:mm:ss');
|
const time = dateFormat(new Date(), 'HH:mm:ss');
|
||||||
const worker = cluster.isPrimary ? '*' : cluster.worker.id;
|
const worker = cluster.isPrimary ? '*' : cluster.worker!.id;
|
||||||
const l =
|
const l =
|
||||||
level === 'error' ? important ? chalk.bgRed.white('ERR ') : chalk.red('ERR ') :
|
level === 'error' ? important ? chalk.bgRed.white('ERR ') : chalk.red('ERR ') :
|
||||||
level === 'warning' ? chalk.yellow('WARN') :
|
level === 'warning' ? chalk.yellow('WARN') :
|
||||||
@@ -68,20 +66,7 @@ export default class Logger {
|
|||||||
if (envOption.withLogTime) log = chalk.gray(time) + ' ' + log;
|
if (envOption.withLogTime) log = chalk.gray(time) + ' ' + log;
|
||||||
|
|
||||||
console.log(important ? chalk.bold(log) : log);
|
console.log(important ? chalk.bold(log) : log);
|
||||||
|
if (level === 'error' && data) console.log(data);
|
||||||
if (store) {
|
|
||||||
if (this.syslogClient) {
|
|
||||||
const send =
|
|
||||||
level === 'error' ? this.syslogClient.error :
|
|
||||||
level === 'warning' ? this.syslogClient.warning :
|
|
||||||
level === 'success' ? this.syslogClient.info :
|
|
||||||
level === 'debug' ? this.syslogClient.info :
|
|
||||||
level === 'info' ? this.syslogClient.info :
|
|
||||||
null as never;
|
|
||||||
|
|
||||||
send.bind(this.syslogClient)(message).catch(() => {});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@bindThis
|
@bindThis
|
||||||
|
@@ -1,5 +1,7 @@
|
|||||||
import { bindThis } from '@/decorators.js';
|
import { bindThis } from '@/decorators.js';
|
||||||
|
|
||||||
|
// TODO: メモリ節約のためあまり参照されないキーを定期的に削除できるようにする?
|
||||||
|
|
||||||
export class Cache<T> {
|
export class Cache<T> {
|
||||||
public cache: Map<string | null, { date: number; value: T; }>;
|
public cache: Map<string | null, { date: number; value: T; }>;
|
||||||
private lifetime: number;
|
private lifetime: number;
|
||||||
|
11
packages/backend/src/misc/dev-null.ts
Normal file
11
packages/backend/src/misc/dev-null.ts
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
import { Writable, WritableOptions } from "node:stream";
|
||||||
|
|
||||||
|
export class DevNull extends Writable implements NodeJS.WritableStream {
|
||||||
|
constructor(opts?: WritableOptions) {
|
||||||
|
super(opts);
|
||||||
|
}
|
||||||
|
|
||||||
|
_write (chunk: any, encoding: BufferEncoding, cb: (err?: Error | null) => void) {
|
||||||
|
setImmediate(cb);
|
||||||
|
}
|
||||||
|
}
|
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user