Compare commits
	
		
			82 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | a103032d94 | ||
|   | c7207a4bd7 | ||
|   | 35c65fe589 | ||
|   | 6d5bd0c484 | ||
|   | cfbb6e8092 | ||
|   | feef4a933e | ||
|   | 468bc67569 | ||
|   | 0d517fa52f | ||
|   | d9054367c1 | ||
|   | 1213373027 | ||
|   | 100a525507 | ||
|   | 1bec4e2d12 | ||
|   | 03cd1d27bf | ||
|   | 9427a756c9 | ||
| ![greenkeeper[bot]](/assets/img/avatar_default.png)  | d32b2a8ce5 | ||
| ![greenkeeper[bot]](/assets/img/avatar_default.png)  | 15473b4368 | ||
|   | 54de0dc4a7 | ||
|   | 0162eaf826 | ||
|   | 572cfafbe1 | ||
|   | 4d6335ce9a | ||
|   | 1c9c4af9f1 | ||
|   | a6844ebc9d | ||
|   | 072492c29b | ||
|   | 99da4f9839 | ||
|   | 88664486af | ||
|   | 80daf7c749 | ||
| ![greenkeeper[bot]](/assets/img/avatar_default.png)  | beb2f7e558 | ||
| ![greenkeeper[bot]](/assets/img/avatar_default.png)  | 6243184c95 | ||
|   | 1b3baef966 | ||
| ![greenkeeper[bot]](/assets/img/avatar_default.png)  | 98f38ee29b | ||
| ![greenkeeper[bot]](/assets/img/avatar_default.png)  | 09b82bfea4 | ||
|   | 937f686264 | ||
|   | 9bc9cbac21 | ||
|   | 6024550158 | ||
|   | 4ae5f82171 | ||
|   | 6d2c9dcee9 | ||
|   | 0f1b0e1870 | ||
|   | 81c682cdc8 | ||
|   | ab9fa67d9f | ||
|   | 9537fce335 | ||
|   | 9d97e7e348 | ||
|   | ebe7939412 | ||
|   | 807e3e8ca7 | ||
|   | a59faf9117 | ||
|   | d786036155 | ||
|   | 61d6ed5489 | ||
|   | b38200d48a | ||
|   | a0c396a842 | ||
|   | 88fbc53e37 | ||
|   | a2206b2d52 | ||
|   | a95ff447d7 | ||
|   | 49dbd7f9d2 | ||
|   | 2ad2779096 | ||
|   | 23045369aa | ||
|   | a13319fd86 | ||
|   | be8765278c | ||
|   | c8bb3dc209 | ||
|   | ea16befb73 | ||
|   | 20b1bb7681 | ||
|   | bd10eb50eb | ||
|   | d47c0eb31a | ||
|   | 177e8bb19f | ||
|   | d156111637 | ||
|   | 8c13d3e50b | ||
|   | 6ff01016f0 | ||
|   | 5d659da012 | ||
|   | 28e7552a1a | ||
|   | 53d264814b | ||
|   | 935b074a7a | ||
|   | 9d9c609bfb | ||
|   | f6a664f181 | ||
|   | fce68d1f75 | ||
|   | 88739c2444 | ||
|   | 7e2f10fce3 | ||
|   | a494c3a5cc | ||
|   | d6bb702883 | ||
|   | d15a972c68 | ||
|   | 2ae7d31725 | ||
|   | 2e329b1888 | ||
|   | 522d40328b | ||
|   | 2ecbff45bf | ||
|   | b6f7282c13 | 
| @@ -30,7 +30,7 @@ while : | ||||
|   touch patreon.cache && \ | ||||
|   rm patreon.cache && \ | ||||
|   cat patreon.raw.cache | \ | ||||
|   jq -r '(.data|map(select(.relationships.currently_entitled_tiers.data[]))|map(.relationships.user.data.id))as$data|.included|map(select(.attributes.hide_pledges==false))|map(select(.id as$id|$data|contains([$id])))|map(.attributes|[.full_name,.thumb_url,.url]|@tsv)|.[]|@text' >> patreon.cache && \ | ||||
|   jq -r '(.data|map(select(.relationships.currently_entitled_tiers.data[]))|map(.relationships.user.data.id))as$data|.included|map(select(.id as$id|$data|contains([$id])))|map(.attributes|[.full_name,.thumb_url,.url]|@tsv)|.[]|@text' >> patreon.cache && \ | ||||
|   echo '<table><tr>' >> patreon.md.cache && \ | ||||
|   cat patreon.cache | \ | ||||
|   awk -F'\t' '{print $2,$1}' | \ | ||||
|   | ||||
| @@ -1,12 +1,8 @@ | ||||
| maintainer: '@syuilo' | ||||
| url: 'https://misskey.xyz' | ||||
| secondary_url: 'https://himasaku.net' | ||||
| maintainer: | ||||
|   name: syuilo | ||||
|   url: 'https://syuilo.com' | ||||
| url: 'http://misskey.local' | ||||
| port: 80 | ||||
| https: | ||||
|   enable: false | ||||
|   key: null | ||||
|   cert: null | ||||
|   ca: null | ||||
| mongodb: | ||||
|   host: localhost | ||||
|   port: 27017 | ||||
| @@ -21,6 +17,3 @@ elasticsearch: | ||||
|   host: localhost | ||||
|   port: 9200 | ||||
|   pass: '' | ||||
| recaptcha: | ||||
|   site_key: hima | ||||
|   secret_key: saku | ||||
|   | ||||
| @@ -1,12 +1,8 @@ | ||||
| maintainer: '@syuilo' | ||||
| url: 'https://misskey.xyz' | ||||
| secondary_url: 'https://himasaku.net' | ||||
| maintainer: | ||||
|   name: syuilo | ||||
|   url: 'https://syuilo.com' | ||||
| url: 'http://misskey.local' | ||||
| port: 80 | ||||
| https: | ||||
|   enable: false | ||||
|   key: null | ||||
|   cert: null | ||||
|   ca: null | ||||
| mongodb: | ||||
|   host: localhost | ||||
|   port: 27017 | ||||
| @@ -21,6 +17,3 @@ elasticsearch: | ||||
|   host: localhost | ||||
|   port: 9200 | ||||
|   pass: '' | ||||
| recaptcha: | ||||
|   site_key: hima | ||||
|   secret_key: saku | ||||
|   | ||||
| @@ -785,6 +785,10 @@ desktop/views/components/settings.vue: | ||||
|   tools: "ツール" | ||||
|   task-manager: "タスクマネージャ" | ||||
|   third-parties: "サードパーティ" | ||||
|   navbar-position: "ナビゲーションバーの位置" | ||||
|   navbar-position-top: "上" | ||||
|   navbar-position-left: "左" | ||||
|   navbar-position-right: "右" | ||||
| desktop/views/components/settings.2fa.vue: | ||||
|   intro: "二段階認証を設定すると、サインイン時にパスワードだけでなく、予め登録しておいた物理的なデバイス(例えばあなたのスマートフォンなど)も必要になり、よりセキュリティが向上します。" | ||||
|   detail: "詳細..." | ||||
| @@ -834,6 +838,7 @@ desktop/views/components/settings.profile.vue: | ||||
|   save: "保存" | ||||
|   locked-account: "アカウントの保護" | ||||
|   is-locked: "フォローを承認制にする" | ||||
|   careful-bot: "Botからのフォローだけ承認制にする" | ||||
|   other: "その他" | ||||
|   is-bot: "このアカウントはBotです" | ||||
|   is-cat: "このアカウントはCatです" | ||||
| @@ -1074,6 +1079,8 @@ mobile/views/components/drive.file-detail.vue: | ||||
|   hash: "ハッシュ (md5)" | ||||
|   exif: "EXIF" | ||||
|   nsfw: "閲覧注意" | ||||
|   mark-as-sensitive: "閲覧注意に設定" | ||||
|   unmark-as-sensitive: "閲覧注意を解除" | ||||
| mobile/views/components/media-image.vue: | ||||
|   sensitive: "閲覧注意" | ||||
|   click-to-show: "クリックして表示" | ||||
| @@ -1225,6 +1232,7 @@ mobile/views/pages/settings/settings.profile.vue: | ||||
|   banner: "バナー" | ||||
|   is-cat: "このアカウントはCatです" | ||||
|   is-locked: "フォローを承認制にする" | ||||
|   careful-bot: "Botからのフォローだけ承認制にする" | ||||
|   advanced: "その他" | ||||
|   privacy: "プライバシー" | ||||
|   save: "保存" | ||||
|   | ||||
| @@ -3,16 +3,16 @@ meta: | ||||
|   lang: "Deutsch" | ||||
|   divider: "" | ||||
| common: | ||||
|   misskey: "A ⭐ of fediverse" | ||||
|   about-title: "A ⭐ of fediverse." | ||||
|   misskey: "Ein ⭐ des Fediversums" | ||||
|   about-title: "Ein ⭐ des Fediversums." | ||||
|   about: "Misskeyを見つけていただき、ありがとうございます。Misskeyは、地球で生まれた<b>分散マイクロブログSNS</b>です。Fediverse(様々なSNSで構成される宇宙)の中に存在するため、他のSNSと相互に繋がっています。暫し都会の喧騒から離れて、新しいインターネットにダイブしてみませんか。" | ||||
|   intro: | ||||
|     title: "Misskeyって?" | ||||
|     title: "Was ist Misskey?" | ||||
|     about: "Misskeyはオープンソースの<b>分散型マイクロブログSNS</b>です。リッチで高度にカスタマイズできるUI、投稿へのリアクション、ファイルを一元管理できるドライブなど、先進的な機能を揃えています。また、Fediverseと呼ばれるネットワークに接続できるため、他のSNSともやり取りできます。例えば、あなたが何か投稿すると、その投稿はMisskeyだけでなく他のSNSにも伝わります。ちょうどある惑星から他の惑星に電波を発信している様子をイメージしてください。" | ||||
|     features: "特徴" | ||||
|     rich-contents: "投稿" | ||||
|     features: "Funktionen" | ||||
|     rich-contents: "Notizen" | ||||
|     rich-contents-desc: "自分の考え、話題の出来事、皆と共有したいことについて発信してください。必要であれば、様々な構文を使って投稿を装飾したり、好きな画像、動画などのファイルやアンケートを添付することもできます。" | ||||
|     reaction: "リアクション" | ||||
|     reaction: "Reaktionen" | ||||
|     reaction-desc: "あなたの気持ちを伝える最も簡単な方法です。Misskeyは、他のユーザーの投稿に様々なリアクションを付けることができます。いちどMisskeyのリアクション機能を体験してしまうと、もう「いいね」の概念しか存在しないSNSには戻れなくなるかもしれません。" | ||||
|     ui: "インターフェース" | ||||
|     ui-desc: "どのようなUIが使いやすいかは人それぞれです。だから、Misskeyは自由度の高いUIを持っています。レイアウトやデザインを調整したり、カスタマイズ可能な様々なウィジェットを配置したりして、自分だけのホームを作ってください。" | ||||
| @@ -23,7 +23,7 @@ common: | ||||
|     detected: "広告ブロッカーを無効にしてください" | ||||
|     warning: "<strong>Misskeyは広告を掲載していません</strong>が、広告をブロックする機能が有効だと一部の機能が利用できなかったり、不具合が発生する場合があります。" | ||||
|   application-authorization: "アプリの連携" | ||||
|   close: "閉じる" | ||||
|   close: "Schließen" | ||||
|   do-not-copy-paste: "ここにコードを入力したり張り付けたりしないでください。アカウントが不正利用される可能性があります。" | ||||
|   got-it: "わかった" | ||||
|   customization-tips: | ||||
| @@ -34,13 +34,13 @@ common: | ||||
|     paragraph4: "カスタマイズを終了するには、右上の「完了」をクリックします。" | ||||
|     gotit: "Got it!" | ||||
|   notification: | ||||
|     file-uploaded: "ファイルがアップロードされました" | ||||
|     file-uploaded: "Datei hochgeladen!" | ||||
|     message-from: "{}さんからメッセージ:" | ||||
|     reversi-invited: "対局への招待があります" | ||||
|     reversi-invited-by: "{}さんから" | ||||
|     notified-by: "{}さんから" | ||||
|     reply-from: "{}さんから返信:" | ||||
|     quoted-by: "{}さんが引用:" | ||||
|     notified-by: "Benachrichtigt von {}:" | ||||
|     reply-from: "Antwort von {}:" | ||||
|     quoted-by: "Zitiert von {}:" | ||||
|   time: | ||||
|     unknown: "Unbekannt" | ||||
|     future: "Zukunft" | ||||
| @@ -53,7 +53,7 @@ common: | ||||
|     months_ago: "vor {0} Monat{0:en}" | ||||
|     years_ago: "vor {} Jahr{0:en}" | ||||
|   month-and-day: "{month}月 {day}日" | ||||
|   trash: "ゴミ箱" | ||||
|   trash: "Papierkorb" | ||||
|   weekday-short: | ||||
|     sunday: "So" | ||||
|     monday: "Mo" | ||||
| @@ -63,13 +63,13 @@ common: | ||||
|     friday: "Fr" | ||||
|     saturday: "Sa" | ||||
|   weekday: | ||||
|     sunday: "日曜日" | ||||
|     monday: "月曜日" | ||||
|     tuesday: "火曜日" | ||||
|     wednesday: "水曜日" | ||||
|     thursday: "木曜日" | ||||
|     friday: "金曜日" | ||||
|     saturday: "土曜日" | ||||
|     sunday: "Sonntag" | ||||
|     monday: "Montag" | ||||
|     tuesday: "Dienstag" | ||||
|     wednesday: "Mittwoch" | ||||
|     thursday: "Donnerstag" | ||||
|     friday: "Freitag" | ||||
|     saturday: "Samstag" | ||||
|   reactions: | ||||
|     like: "いいね" | ||||
|     love: "Lieben" | ||||
| @@ -82,10 +82,10 @@ common: | ||||
|     rip: "RIP" | ||||
|     pudding: "Pudding" | ||||
|   note-visibility: | ||||
|     public: "公開" | ||||
|     public: "Öffentlich" | ||||
|     home: "ホーム" | ||||
|     home-desc: "ホームタイムラインにのみ公開" | ||||
|     followers: "フォロワー" | ||||
|     followers: "Abonnenten" | ||||
|     followers-desc: "自分のフォロワーにのみ公開" | ||||
|     specified: "ダイレクト" | ||||
|     specified-desc: "指定したユーザーにのみ公開" | ||||
| @@ -122,9 +122,9 @@ common: | ||||
|     turn-of: "{}のターンです" | ||||
|     past-turn-of: "{}のターン" | ||||
|     won: "{}の勝ち" | ||||
|     black: "黒" | ||||
|     white: "白" | ||||
|     total: "合計" | ||||
|     black: "Schwarz" | ||||
|     white: "Weiß" | ||||
|     total: "Gesamt" | ||||
|     this-turn: "{}ターン目" | ||||
|   widgets: | ||||
|     analog-clock: "Analoge Uhr" | ||||
| @@ -142,23 +142,23 @@ common: | ||||
|     broadcast: "ブロードキャスト" | ||||
|     notifications: "Benachrichtigungen" | ||||
|     users: "Empfohlene Benutzer" | ||||
|     polls: "アンケート" | ||||
|     polls: "Umfrage" | ||||
|     post-form: "Beitragsform" | ||||
|     messaging: "Nachrichten" | ||||
|     server: "Server-Info" | ||||
|     donation: "Spenden" | ||||
|     nav: "Navigation" | ||||
|     tips: "Tipps" | ||||
|     hashtags: "ハッシュタグ" | ||||
|     hashtags: "Hashtags" | ||||
|   deck: | ||||
|     widgets: "Widget hinzufügen:" | ||||
|     home: "Startseite" | ||||
|     local: "Lokal" | ||||
|     hybrid: "ソーシャル" | ||||
|     hashtag: "ハッシュタグ" | ||||
|     hashtag: "Hashtag" | ||||
|     global: "Global" | ||||
|     mentions: "あなた宛て" | ||||
|     direct: "ダイレクト投稿" | ||||
|     mentions: "Erwähnungen" | ||||
|     direct: "Direktnachrichten" | ||||
|     notifications: "Mitteilungen" | ||||
|     list: "Listen" | ||||
|     swap-left: "Nach links" | ||||
| @@ -182,10 +182,10 @@ auth/views/form.vue: | ||||
|   drive-write: "ドライブを操作する。" | ||||
|   notification-read: "通知を見る。" | ||||
|   notification-write: "通知を操作する。" | ||||
|   cancel: "キャンセル" | ||||
|   accept: "アクセスを許可" | ||||
|   cancel: "Abbrechen" | ||||
|   accept: "Zugriff erlauben." | ||||
| auth/views/index.vue: | ||||
|   loading: "読み込み中" | ||||
|   loading: "Lädt" | ||||
|   denied: "アプリケーションの連携をキャンセルしました。" | ||||
|   denied-paragraph: "このアプリがあなたのアカウントにアクセスすることはありません。" | ||||
|   already-authorized: "このアプリは既に連携済みです" | ||||
| @@ -196,10 +196,10 @@ auth/views/index.vue: | ||||
|   sign-in: "サインインしてください" | ||||
| common/views/components/games/reversi/reversi.vue: | ||||
|   matching: | ||||
|     waiting-for: "{}を待っています" | ||||
|     cancel: "キャンセル" | ||||
|     waiting-for: "Warten auf {}" | ||||
|     cancel: "Abbrechen" | ||||
| common/views/components/games/reversi/reversi.game.vue: | ||||
|   surrender: "投了" | ||||
|   surrender: "Aufgeben" | ||||
|   surrendered: "投了により" | ||||
|   is-llotheo: "石の少ない方が勝ち(ロセオ)" | ||||
|   looped-map: "ループマップ" | ||||
| @@ -208,9 +208,9 @@ common/views/components/games/reversi/reversi.index.vue: | ||||
|   title: "Misskey Reversi" | ||||
|   sub-title: "他のMisskeyユーザーとリバーシで対戦しよう" | ||||
|   invite: "招待" | ||||
|   rule: "遊び方" | ||||
|   rule: "Spielanleitung" | ||||
|   rule-desc: "リバーシは、相手と交互に石をボードに置いて、相手の石を挟んで自分の色に変えてゆき、最終的に残った石が多い方が勝ちというボードゲームです。" | ||||
|   mode-invite: "招待" | ||||
|   mode-invite: "Einladen" | ||||
|   mode-invite-desc: "指定したユーザーと対戦するモードです。" | ||||
|   invitations: "対局の招待があります!" | ||||
|   my-games: "自分の対局" | ||||
| @@ -785,6 +785,10 @@ desktop/views/components/settings.vue: | ||||
|   tools: "Werkzeuge" | ||||
|   task-manager: "Taskmanager" | ||||
|   third-parties: "サードパーティ" | ||||
|   navbar-position: "ナビゲーションバーの位置" | ||||
|   navbar-position-top: "上" | ||||
|   navbar-position-left: "左" | ||||
|   navbar-position-right: "右" | ||||
| desktop/views/components/settings.2fa.vue: | ||||
|   intro: "二段階認証を設定すると、サインイン時にパスワードだけでなく、予め登録しておいた物理的なデバイス(例えばあなたのスマートフォンなど)も必要になり、よりセキュリティが向上します。" | ||||
|   detail: "詳細..." | ||||
| @@ -834,6 +838,7 @@ desktop/views/components/settings.profile.vue: | ||||
|   save: "Profil aktualisieren" | ||||
|   locked-account: "アカウントの保護" | ||||
|   is-locked: "フォローを承認制にする" | ||||
|   careful-bot: "Botからのフォローだけ承認制にする" | ||||
|   other: "その他" | ||||
|   is-bot: "このアカウントはBotです" | ||||
|   is-cat: "このアカウントはCatです" | ||||
| @@ -1074,6 +1079,8 @@ mobile/views/components/drive.file-detail.vue: | ||||
|   hash: "ハッシュ (md5)" | ||||
|   exif: "EXIF" | ||||
|   nsfw: "閲覧注意" | ||||
|   mark-as-sensitive: "閲覧注意に設定" | ||||
|   unmark-as-sensitive: "閲覧注意を解除" | ||||
| mobile/views/components/media-image.vue: | ||||
|   sensitive: "閲覧注意" | ||||
|   click-to-show: "クリックして表示" | ||||
| @@ -1225,6 +1232,7 @@ mobile/views/pages/settings/settings.profile.vue: | ||||
|   banner: "バナー" | ||||
|   is-cat: "このアカウントはCatです" | ||||
|   is-locked: "フォローを承認制にする" | ||||
|   careful-bot: "Botからのフォローだけ承認制にする" | ||||
|   advanced: "その他" | ||||
|   privacy: "プライバシー" | ||||
|   save: "保存" | ||||
|   | ||||
| @@ -785,6 +785,10 @@ desktop/views/components/settings.vue: | ||||
|   tools: "Tools" | ||||
|   task-manager: "Task Manager" | ||||
|   third-parties: "Third-parties" | ||||
|   navbar-position: "Navigation bar position" | ||||
|   navbar-position-top: "Top" | ||||
|   navbar-position-left: "Left" | ||||
|   navbar-position-right: "Right" | ||||
| desktop/views/components/settings.2fa.vue: | ||||
|   intro: "If you set up 2-step verification, you will not only need a password at sign-in, but also a pre-registered physical device (such as your smartphone), which will improve security." | ||||
|   detail: "Details…" | ||||
| @@ -834,6 +838,7 @@ desktop/views/components/settings.profile.vue: | ||||
|   save: "Update profile" | ||||
|   locked-account: "Protect your account" | ||||
|   is-locked: "Follow request needs approval" | ||||
|   careful-bot: "Botからのフォローだけ承認制にする" | ||||
|   other: "Other" | ||||
|   is-bot: "This account is a Bot" | ||||
|   is-cat: "This account is a Cat" | ||||
| @@ -1074,6 +1079,8 @@ mobile/views/components/drive.file-detail.vue: | ||||
|   hash: "Hash (md5)" | ||||
|   exif: "EXIF" | ||||
|   nsfw: "NSFW" | ||||
|   mark-as-sensitive: "Mark as 'sensitive'" | ||||
|   unmark-as-sensitive: "Unmark as 'sensitive'" | ||||
| mobile/views/components/media-image.vue: | ||||
|   sensitive: "NSFW" | ||||
|   click-to-show: "Click to show" | ||||
| @@ -1225,6 +1232,7 @@ mobile/views/pages/settings/settings.profile.vue: | ||||
|   banner: "Banner" | ||||
|   is-cat: "This account is a Cat" | ||||
|   is-locked: "Follow request needs approval" | ||||
|   careful-bot: "Botからのフォローだけ承認制にする" | ||||
|   advanced: "Advanced" | ||||
|   privacy: "Privacy" | ||||
|   save: "Update profile" | ||||
|   | ||||
| @@ -785,6 +785,10 @@ desktop/views/components/settings.vue: | ||||
|   tools: "Herramientas" | ||||
|   task-manager: "Navegador de tareas" | ||||
|   third-parties: "Servicios externos" | ||||
|   navbar-position: "ナビゲーションバーの位置" | ||||
|   navbar-position-top: "上" | ||||
|   navbar-position-left: "左" | ||||
|   navbar-position-right: "右" | ||||
| desktop/views/components/settings.2fa.vue: | ||||
|   intro: "二段階認証を設定すると、サインイン時にパスワードだけでなく、予め登録しておいた物理的なデバイス(例えばあなたのスマートフォンなど)も必要になり、よりセキュリティが向上します。" | ||||
|   detail: "Ver detalles..." | ||||
| @@ -834,6 +838,7 @@ desktop/views/components/settings.profile.vue: | ||||
|   save: "Perfil actualizado" | ||||
|   locked-account: "Protege tu cuenta" | ||||
|   is-locked: "フォローを承認制にする" | ||||
|   careful-bot: "Botからのフォローだけ承認制にする" | ||||
|   other: "その他" | ||||
|   is-bot: "このアカウントはBotです" | ||||
|   is-cat: "このアカウントはCatです" | ||||
| @@ -1074,6 +1079,8 @@ mobile/views/components/drive.file-detail.vue: | ||||
|   hash: "ハッシュ (md5)" | ||||
|   exif: "EXIF" | ||||
|   nsfw: "閲覧注意" | ||||
|   mark-as-sensitive: "閲覧注意に設定" | ||||
|   unmark-as-sensitive: "閲覧注意を解除" | ||||
| mobile/views/components/media-image.vue: | ||||
|   sensitive: "閲覧注意" | ||||
|   click-to-show: "クリックして表示" | ||||
| @@ -1225,6 +1232,7 @@ mobile/views/pages/settings/settings.profile.vue: | ||||
|   banner: "バナー" | ||||
|   is-cat: "このアカウントはCatです" | ||||
|   is-locked: "フォローを承認制にする" | ||||
|   careful-bot: "Botからのフォローだけ承認制にする" | ||||
|   advanced: "その他" | ||||
|   privacy: "プライバシー" | ||||
|   save: "保存" | ||||
|   | ||||
| @@ -785,6 +785,10 @@ desktop/views/components/settings.vue: | ||||
|   tools: "Outils" | ||||
|   task-manager: "Gestionnaire de tâches" | ||||
|   third-parties: "Services tiers" | ||||
|   navbar-position: "ナビゲーションバーの位置" | ||||
|   navbar-position-top: "上" | ||||
|   navbar-position-left: "左" | ||||
|   navbar-position-right: "右" | ||||
| desktop/views/components/settings.2fa.vue: | ||||
|   intro: "Si vous configurez la vérication en deux étapes vous aurez non seulement besoin de votre mot de passe mais aussi un appareil déjà pré-enregistré(tel que votre smartphone) ce qui ameliora grandement la sécurité de votre compte." | ||||
|   detail: "Voir les détails..." | ||||
| @@ -834,6 +838,7 @@ desktop/views/components/settings.profile.vue: | ||||
|   save: "Mettre à jour le profil" | ||||
|   locked-account: "Protéger votre compte" | ||||
|   is-locked: "Demande d’abonnement en attente d’approbation" | ||||
|   careful-bot: "Botからのフォローだけ承認制にする" | ||||
|   other: "Autre" | ||||
|   is-bot: "Ce compte est un Bot" | ||||
|   is-cat: "Ce compte est un Chat" | ||||
| @@ -1074,6 +1079,8 @@ mobile/views/components/drive.file-detail.vue: | ||||
|   hash: "Hash (md5)" | ||||
|   exif: "EXIF" | ||||
|   nsfw: "CW" | ||||
|   mark-as-sensitive: "閲覧注意に設定" | ||||
|   unmark-as-sensitive: "閲覧注意を解除" | ||||
| mobile/views/components/media-image.vue: | ||||
|   sensitive: "Le contenu est NSFW" | ||||
|   click-to-show: "Cliquer pour afficher" | ||||
| @@ -1225,6 +1232,7 @@ mobile/views/pages/settings/settings.profile.vue: | ||||
|   banner: "Bannière" | ||||
|   is-cat: "Ce compte est un Bot" | ||||
|   is-locked: "Demande d’abonnement en attente d’approbation" | ||||
|   careful-bot: "Botからのフォローだけ承認制にする" | ||||
|   advanced: "Avancé" | ||||
|   privacy: "Vie privée" | ||||
|   save: "Mettre à jour le profil" | ||||
|   | ||||
| @@ -785,6 +785,10 @@ desktop/views/components/settings.vue: | ||||
|   tools: "ツール" | ||||
|   task-manager: "タスクマネージャ" | ||||
|   third-parties: "サードパーティ" | ||||
|   navbar-position: "ナビゲーションバーの位置" | ||||
|   navbar-position-top: "上" | ||||
|   navbar-position-left: "左" | ||||
|   navbar-position-right: "右" | ||||
| desktop/views/components/settings.2fa.vue: | ||||
|   intro: "二段階認証を設定すると、サインイン時にパスワードだけでなく、予め登録しておいた物理的なデバイス(例えばあなたのスマートフォンなど)も必要になり、よりセキュリティが向上します。" | ||||
|   detail: "詳細..." | ||||
| @@ -834,6 +838,7 @@ desktop/views/components/settings.profile.vue: | ||||
|   save: "保存" | ||||
|   locked-account: "アカウントの保護" | ||||
|   is-locked: "フォローを承認制にする" | ||||
|   careful-bot: "Botからのフォローだけ承認制にする" | ||||
|   other: "その他" | ||||
|   is-bot: "このアカウントはBotです" | ||||
|   is-cat: "このアカウントはCatです" | ||||
| @@ -1074,6 +1079,8 @@ mobile/views/components/drive.file-detail.vue: | ||||
|   hash: "ハッシュ (md5)" | ||||
|   exif: "EXIF" | ||||
|   nsfw: "閲覧注意" | ||||
|   mark-as-sensitive: "閲覧注意に設定" | ||||
|   unmark-as-sensitive: "閲覧注意を解除" | ||||
| mobile/views/components/media-image.vue: | ||||
|   sensitive: "閲覧注意" | ||||
|   click-to-show: "クリックして表示" | ||||
| @@ -1225,6 +1232,7 @@ mobile/views/pages/settings/settings.profile.vue: | ||||
|   banner: "バナー" | ||||
|   is-cat: "このアカウントはCatです" | ||||
|   is-locked: "フォローを承認制にする" | ||||
|   careful-bot: "Botからのフォローだけ承認制にする" | ||||
|   advanced: "その他" | ||||
|   privacy: "プライバシー" | ||||
|   save: "保存" | ||||
|   | ||||
| @@ -1239,6 +1239,8 @@ mobile/views/components/drive.file-detail.vue: | ||||
|   hash: "ハッシュ (md5)" | ||||
|   exif: "EXIF" | ||||
|   nsfw: "閲覧注意" | ||||
|   mark-as-sensitive: "閲覧注意に設定" | ||||
|   unmark-as-sensitive: "閲覧注意を解除" | ||||
|  | ||||
| mobile/views/components/media-image.vue: | ||||
|   sensitive: "閲覧注意" | ||||
|   | ||||
| @@ -785,6 +785,10 @@ desktop/views/components/settings.vue: | ||||
|   tools: "ツール" | ||||
|   task-manager: "タスクマネージャ" | ||||
|   third-parties: "サードパーティ" | ||||
|   navbar-position: "ナビゲーションバーの位置" | ||||
|   navbar-position-top: "上" | ||||
|   navbar-position-left: "左" | ||||
|   navbar-position-right: "右" | ||||
| desktop/views/components/settings.2fa.vue: | ||||
|   intro: "二段階認証を設定すると、サインイン時にパスワードだけとちゃうくて、予め登録しておいた物理的なデバイス(例えばあんさんのスマートフォンなど)も必要になり、よりセキュリティが向上すんで。" | ||||
|   detail: "詳細..." | ||||
| @@ -834,6 +838,7 @@ desktop/views/components/settings.profile.vue: | ||||
|   save: "保存" | ||||
|   locked-account: "アカウント守る" | ||||
|   is-locked: "他人のフォローは許可してからや!" | ||||
|   careful-bot: "Botからのフォローだけ承認制にする" | ||||
|   other: "その他" | ||||
|   is-bot: "このアカウントはBotやで" | ||||
|   is-cat: "このアカウントはCatやで" | ||||
| @@ -1074,6 +1079,8 @@ mobile/views/components/drive.file-detail.vue: | ||||
|   hash: "ハッシュ(md5)" | ||||
|   exif: "EXIF" | ||||
|   nsfw: "ちょっと見せられへんわ" | ||||
|   mark-as-sensitive: "閲覧注意に設定" | ||||
|   unmark-as-sensitive: "閲覧注意を解除" | ||||
| mobile/views/components/media-image.vue: | ||||
|   sensitive: "見たらあかんで" | ||||
|   click-to-show: "押してみ、見せたるわ" | ||||
| @@ -1225,6 +1232,7 @@ mobile/views/pages/settings/settings.profile.vue: | ||||
|   banner: "バナー" | ||||
|   is-cat: "このアカウントはCatや" | ||||
|   is-locked: "他人のフォローは許可してからや!" | ||||
|   careful-bot: "Botからのフォローだけ承認制にする" | ||||
|   advanced: "その他" | ||||
|   privacy: "プライバシーってなんや?オカンの年齢か?" | ||||
|   save: "保存" | ||||
|   | ||||
| @@ -785,6 +785,10 @@ desktop/views/components/settings.vue: | ||||
|   tools: "ツール" | ||||
|   task-manager: "タスクマネージャ" | ||||
|   third-parties: "サードパーティ" | ||||
|   navbar-position: "ナビゲーションバーの位置" | ||||
|   navbar-position-top: "上" | ||||
|   navbar-position-left: "左" | ||||
|   navbar-position-right: "右" | ||||
| desktop/views/components/settings.2fa.vue: | ||||
|   intro: "二段階認証を設定すると、サインイン時にパスワードだけでなく、予め登録しておいた物理的なデバイス(例えばあなたのスマートフォンなど)も必要になり、よりセキュリティが向上します。" | ||||
|   detail: "詳細..." | ||||
| @@ -834,6 +838,7 @@ desktop/views/components/settings.profile.vue: | ||||
|   save: "保存" | ||||
|   locked-account: "アカウントの保護" | ||||
|   is-locked: "フォローを承認制にする" | ||||
|   careful-bot: "Botからのフォローだけ承認制にする" | ||||
|   other: "その他" | ||||
|   is-bot: "このアカウントはBotです" | ||||
|   is-cat: "このアカウントはCatです" | ||||
| @@ -1074,6 +1079,8 @@ mobile/views/components/drive.file-detail.vue: | ||||
|   hash: "ハッシュ (md5)" | ||||
|   exif: "EXIF" | ||||
|   nsfw: "閲覧注意" | ||||
|   mark-as-sensitive: "閲覧注意に設定" | ||||
|   unmark-as-sensitive: "閲覧注意を解除" | ||||
| mobile/views/components/media-image.vue: | ||||
|   sensitive: "閲覧注意" | ||||
|   click-to-show: "クリックして表示" | ||||
| @@ -1225,6 +1232,7 @@ mobile/views/pages/settings/settings.profile.vue: | ||||
|   banner: "バナー" | ||||
|   is-cat: "このアカウントはCatです" | ||||
|   is-locked: "フォローを承認制にする" | ||||
|   careful-bot: "Botからのフォローだけ承認制にする" | ||||
|   advanced: "その他" | ||||
|   privacy: "プライバシー" | ||||
|   save: "保存" | ||||
|   | ||||
| @@ -785,6 +785,10 @@ desktop/views/components/settings.vue: | ||||
|   tools: "Hulpmiddelen" | ||||
|   task-manager: "Taakbeheer" | ||||
|   third-parties: "Derde partij" | ||||
|   navbar-position: "ナビゲーションバーの位置" | ||||
|   navbar-position-top: "上" | ||||
|   navbar-position-left: "左" | ||||
|   navbar-position-right: "右" | ||||
| desktop/views/components/settings.2fa.vue: | ||||
|   intro: "Als je verificatie in twee stappen instelt, dan heb je niet alleen een wachtwoord nodig bij het inloggen, maar ook een geregistreerd fysiek apparaat (zoals je smartphone). Dit verhoogt de veiligheid. " | ||||
|   detail: "Details bekijken..." | ||||
| @@ -834,6 +838,7 @@ desktop/views/components/settings.profile.vue: | ||||
|   save: "Profiel bijwerken" | ||||
|   locked-account: "アカウントの保護" | ||||
|   is-locked: "フォローを承認制にする" | ||||
|   careful-bot: "Botからのフォローだけ承認制にする" | ||||
|   other: "その他" | ||||
|   is-bot: "Dit account is een Bot" | ||||
|   is-cat: "Dit account is een Kat" | ||||
| @@ -1074,6 +1079,8 @@ mobile/views/components/drive.file-detail.vue: | ||||
|   hash: "Hash (md5)" | ||||
|   exif: "EXIF" | ||||
|   nsfw: "閲覧注意" | ||||
|   mark-as-sensitive: "閲覧注意に設定" | ||||
|   unmark-as-sensitive: "閲覧注意を解除" | ||||
| mobile/views/components/media-image.vue: | ||||
|   sensitive: "閲覧注意" | ||||
|   click-to-show: "クリックして表示" | ||||
| @@ -1225,6 +1232,7 @@ mobile/views/pages/settings/settings.profile.vue: | ||||
|   banner: "Omslagfoto" | ||||
|   is-cat: "Dit account is een Kat" | ||||
|   is-locked: "フォローを承認制にする" | ||||
|   careful-bot: "Botからのフォローだけ承認制にする" | ||||
|   advanced: "その他" | ||||
|   privacy: "プライバシー" | ||||
|   save: "Profiel bijwerken" | ||||
|   | ||||
| @@ -785,6 +785,10 @@ desktop/views/components/settings.vue: | ||||
|   tools: "Verktøy" | ||||
|   task-manager: "タスクマネージャ" | ||||
|   third-parties: "サードパーティ" | ||||
|   navbar-position: "ナビゲーションバーの位置" | ||||
|   navbar-position-top: "上" | ||||
|   navbar-position-left: "左" | ||||
|   navbar-position-right: "右" | ||||
| desktop/views/components/settings.2fa.vue: | ||||
|   intro: "二段階認証を設定すると、サインイン時にパスワードだけでなく、予め登録しておいた物理的なデバイス(例えばあなたのスマートフォンなど)も必要になり、よりセキュリティが向上します。" | ||||
|   detail: "Detaljer..." | ||||
| @@ -834,6 +838,7 @@ desktop/views/components/settings.profile.vue: | ||||
|   save: "Lagre profilen" | ||||
|   locked-account: "アカウントの保護" | ||||
|   is-locked: "フォローを承認制にする" | ||||
|   careful-bot: "Botからのフォローだけ承認制にする" | ||||
|   other: "Annet" | ||||
|   is-bot: "このアカウントはBotです" | ||||
|   is-cat: "このアカウントはCatです" | ||||
| @@ -1074,6 +1079,8 @@ mobile/views/components/drive.file-detail.vue: | ||||
|   hash: "ハッシュ (md5)" | ||||
|   exif: "EXIF" | ||||
|   nsfw: "NSFW" | ||||
|   mark-as-sensitive: "閲覧注意に設定" | ||||
|   unmark-as-sensitive: "閲覧注意を解除" | ||||
| mobile/views/components/media-image.vue: | ||||
|   sensitive: "NSFW" | ||||
|   click-to-show: "クリックして表示" | ||||
| @@ -1225,6 +1232,7 @@ mobile/views/pages/settings/settings.profile.vue: | ||||
|   banner: "Banner" | ||||
|   is-cat: "このアカウントはCatです" | ||||
|   is-locked: "フォローを承認制にする" | ||||
|   careful-bot: "Botからのフォローだけ承認制にする" | ||||
|   advanced: "Avansert" | ||||
|   privacy: "プライバシー" | ||||
|   save: "Lagre profilen" | ||||
|   | ||||
| @@ -785,6 +785,10 @@ desktop/views/components/settings.vue: | ||||
|   tools: "Narzędzia" | ||||
|   task-manager: "Menedżer zadań" | ||||
|   third-parties: "Autorzy trzeci" | ||||
|   navbar-position: "ナビゲーションバーの位置" | ||||
|   navbar-position-top: "上" | ||||
|   navbar-position-left: "左" | ||||
|   navbar-position-right: "右" | ||||
| desktop/views/components/settings.2fa.vue: | ||||
|   intro: "Jeżeli skonfigurujesz uwierzytelnianie dwuetapowe, aby zablokować się będziesz potrzebować (oprócz hasła) kodu ze skonfigurowanego urządzenia (np. smartfonu), co zwiększy bezpieczeństwo." | ||||
|   detail: "Zobacz szczegóły…" | ||||
| @@ -834,6 +838,7 @@ desktop/views/components/settings.profile.vue: | ||||
|   save: "Aktualizuj profil" | ||||
|   locked-account: "Zabezpiecz swoje konto" | ||||
|   is-locked: "フォローを承認制にする" | ||||
|   careful-bot: "Botからのフォローだけ承認制にする" | ||||
|   other: "Inne" | ||||
|   is-bot: "To konto jest prowadzone przez bota" | ||||
|   is-cat: "To konto jest prowadzone przez kota" | ||||
| @@ -1074,6 +1079,8 @@ mobile/views/components/drive.file-detail.vue: | ||||
|   hash: "Hash (md5)" | ||||
|   exif: "EXIF" | ||||
|   nsfw: "閲覧注意" | ||||
|   mark-as-sensitive: "閲覧注意に設定" | ||||
|   unmark-as-sensitive: "閲覧注意を解除" | ||||
| mobile/views/components/media-image.vue: | ||||
|   sensitive: "To jest zawartość NSFW" | ||||
|   click-to-show: "Naciśnij aby wyświetlić" | ||||
| @@ -1225,6 +1232,7 @@ mobile/views/pages/settings/settings.profile.vue: | ||||
|   banner: "Baner" | ||||
|   is-cat: "To konto jest prowadzone przez kota" | ||||
|   is-locked: "フォローを承認制にする" | ||||
|   careful-bot: "Botからのフォローだけ承認制にする" | ||||
|   advanced: "その他" | ||||
|   privacy: "プライバシー" | ||||
|   save: "Aktualizuj profil" | ||||
|   | ||||
| @@ -785,6 +785,10 @@ desktop/views/components/settings.vue: | ||||
|   tools: "ツール" | ||||
|   task-manager: "タスクマネージャ" | ||||
|   third-parties: "サードパーティ" | ||||
|   navbar-position: "ナビゲーションバーの位置" | ||||
|   navbar-position-top: "上" | ||||
|   navbar-position-left: "左" | ||||
|   navbar-position-right: "右" | ||||
| desktop/views/components/settings.2fa.vue: | ||||
|   intro: "二段階認証を設定すると、サインイン時にパスワードだけでなく、予め登録しておいた物理的なデバイス(例えばあなたのスマートフォンなど)も必要になり、よりセキュリティが向上します。" | ||||
|   detail: "詳細..." | ||||
| @@ -834,6 +838,7 @@ desktop/views/components/settings.profile.vue: | ||||
|   save: "保存" | ||||
|   locked-account: "アカウントの保護" | ||||
|   is-locked: "フォローを承認制にする" | ||||
|   careful-bot: "Botからのフォローだけ承認制にする" | ||||
|   other: "その他" | ||||
|   is-bot: "このアカウントはBotです" | ||||
|   is-cat: "このアカウントはCatです" | ||||
| @@ -1074,6 +1079,8 @@ mobile/views/components/drive.file-detail.vue: | ||||
|   hash: "ハッシュ (md5)" | ||||
|   exif: "EXIF" | ||||
|   nsfw: "閲覧注意" | ||||
|   mark-as-sensitive: "閲覧注意に設定" | ||||
|   unmark-as-sensitive: "閲覧注意を解除" | ||||
| mobile/views/components/media-image.vue: | ||||
|   sensitive: "閲覧注意" | ||||
|   click-to-show: "クリックして表示" | ||||
| @@ -1225,6 +1232,7 @@ mobile/views/pages/settings/settings.profile.vue: | ||||
|   banner: "Capa" | ||||
|   is-cat: "Esta conta é gato" | ||||
|   is-locked: "Pedido para seguir precisa ser aprovado" | ||||
|   careful-bot: "Botからのフォローだけ承認制にする" | ||||
|   advanced: "Avançado" | ||||
|   privacy: "Provacidade" | ||||
|   save: "Atualizar perfil" | ||||
|   | ||||
| @@ -3,9 +3,9 @@ meta: | ||||
|   lang: "Русский язык" | ||||
|   divider: "" | ||||
| common: | ||||
|   misskey: "A ⭐ of fediverse" | ||||
|   about-title: "A ⭐ of fediverse." | ||||
|   about: "Misskeyを見つけていただき、ありがとうございます。Misskeyは、地球で生まれた<b>分散マイクロブログSNS</b>です。Fediverse(様々なSNSで構成される宇宙)の中に存在するため、他のSNSと相互に繋がっています。暫し都会の喧騒から離れて、新しいインターネットにダイブしてみませんか。" | ||||
|   misskey: "Мы — ⭐ fediverse" | ||||
|   about-title: "Мы — ⭐ fediverse" | ||||
|   about: "Спасибо, что нашли Misskey. Misskey — это <b>децентрализованная платформа для микроблоггинга</b> родом с планеты Земля. Поскольку она существует внутри Fediverse (вселенной различных социальных платформ), она связана с другими платформами. Отдохните от шума большого города — и познакомьтесь с новым интернетом." | ||||
|   intro: | ||||
|     title: "Misskeyって?" | ||||
|     about: "Misskeyはオープンソースの<b>分散型マイクロブログSNS</b>です。リッチで高度にカスタマイズできるUI、投稿へのリアクション、ファイルを一元管理できるドライブなど、先進的な機能を揃えています。また、Fediverseと呼ばれるネットワークに接続できるため、他のSNSともやり取りできます。例えば、あなたが何か投稿すると、その投稿はMisskeyだけでなく他のSNSにも伝わります。ちょうどある惑星から他の惑星に電波を発信している様子をイメージしてください。" | ||||
| @@ -785,6 +785,10 @@ desktop/views/components/settings.vue: | ||||
|   tools: "ツール" | ||||
|   task-manager: "タスクマネージャ" | ||||
|   third-parties: "サードパーティ" | ||||
|   navbar-position: "ナビゲーションバーの位置" | ||||
|   navbar-position-top: "上" | ||||
|   navbar-position-left: "左" | ||||
|   navbar-position-right: "右" | ||||
| desktop/views/components/settings.2fa.vue: | ||||
|   intro: "二段階認証を設定すると、サインイン時にパスワードだけでなく、予め登録しておいた物理的なデバイス(例えばあなたのスマートフォンなど)も必要になり、よりセキュリティが向上します。" | ||||
|   detail: "詳細..." | ||||
| @@ -834,6 +838,7 @@ desktop/views/components/settings.profile.vue: | ||||
|   save: "保存" | ||||
|   locked-account: "アカウントの保護" | ||||
|   is-locked: "フォローを承認制にする" | ||||
|   careful-bot: "Botからのフォローだけ承認制にする" | ||||
|   other: "その他" | ||||
|   is-bot: "このアカウントはBotです" | ||||
|   is-cat: "このアカウントはCatです" | ||||
| @@ -1074,6 +1079,8 @@ mobile/views/components/drive.file-detail.vue: | ||||
|   hash: "ハッシュ (md5)" | ||||
|   exif: "EXIF" | ||||
|   nsfw: "閲覧注意" | ||||
|   mark-as-sensitive: "閲覧注意に設定" | ||||
|   unmark-as-sensitive: "閲覧注意を解除" | ||||
| mobile/views/components/media-image.vue: | ||||
|   sensitive: "閲覧注意" | ||||
|   click-to-show: "クリックして表示" | ||||
| @@ -1225,6 +1232,7 @@ mobile/views/pages/settings/settings.profile.vue: | ||||
|   banner: "バナー" | ||||
|   is-cat: "このアカウントはCatです" | ||||
|   is-locked: "フォローを承認制にする" | ||||
|   careful-bot: "Botからのフォローだけ承認制にする" | ||||
|   advanced: "その他" | ||||
|   privacy: "プライバシー" | ||||
|   save: "保存" | ||||
|   | ||||
| @@ -785,6 +785,10 @@ desktop/views/components/settings.vue: | ||||
|   tools: "ツール" | ||||
|   task-manager: "タスクマネージャ" | ||||
|   third-parties: "サードパーティ" | ||||
|   navbar-position: "ナビゲーションバーの位置" | ||||
|   navbar-position-top: "上" | ||||
|   navbar-position-left: "左" | ||||
|   navbar-position-right: "右" | ||||
| desktop/views/components/settings.2fa.vue: | ||||
|   intro: "二段階認証を設定すると、サインイン時にパスワードだけでなく、予め登録しておいた物理的なデバイス(例えばあなたのスマートフォンなど)も必要になり、よりセキュリティが向上します。" | ||||
|   detail: "詳細..." | ||||
| @@ -834,6 +838,7 @@ desktop/views/components/settings.profile.vue: | ||||
|   save: "保存" | ||||
|   locked-account: "アカウントの保護" | ||||
|   is-locked: "フォローを承認制にする" | ||||
|   careful-bot: "Botからのフォローだけ承認制にする" | ||||
|   other: "その他" | ||||
|   is-bot: "このアカウントはBotです" | ||||
|   is-cat: "このアカウントはCatです" | ||||
| @@ -1074,6 +1079,8 @@ mobile/views/components/drive.file-detail.vue: | ||||
|   hash: "ハッシュ (md5)" | ||||
|   exif: "EXIF" | ||||
|   nsfw: "閲覧注意" | ||||
|   mark-as-sensitive: "閲覧注意に設定" | ||||
|   unmark-as-sensitive: "閲覧注意を解除" | ||||
| mobile/views/components/media-image.vue: | ||||
|   sensitive: "閲覧注意" | ||||
|   click-to-show: "クリックして表示" | ||||
| @@ -1225,6 +1232,7 @@ mobile/views/pages/settings/settings.profile.vue: | ||||
|   banner: "バナー" | ||||
|   is-cat: "このアカウントはCatです" | ||||
|   is-locked: "フォローを承認制にする" | ||||
|   careful-bot: "Botからのフォローだけ承認制にする" | ||||
|   advanced: "その他" | ||||
|   privacy: "プライバシー" | ||||
|   save: "保存" | ||||
|   | ||||
							
								
								
									
										29
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										29
									
								
								package.json
									
									
									
									
									
								
							| @@ -1,8 +1,8 @@ | ||||
| { | ||||
| 	"name": "misskey", | ||||
| 	"author": "syuilo <i@syuilo.com>", | ||||
| 	"version": "10.17.0", | ||||
| 	"clientVersion": "1.0.10536", | ||||
| 	"version": "10.21.1", | ||||
| 	"clientVersion": "1.0.10618", | ||||
| 	"codename": "nighthike", | ||||
| 	"main": "./built/index.js", | ||||
| 	"private": true, | ||||
| @@ -20,14 +20,15 @@ | ||||
| 		"format": "gulp format" | ||||
| 	}, | ||||
| 	"dependencies": { | ||||
| 		"@fortawesome/fontawesome-svg-core": "1.2.4", | ||||
| 		"@fortawesome/free-brands-svg-icons": "5.3.1", | ||||
| 		"@fortawesome/free-regular-svg-icons": "5.3.1", | ||||
| 		"@fortawesome/free-solid-svg-icons": "5.3.1", | ||||
| 		"@fortawesome/fontawesome-svg-core": "1.2.6", | ||||
| 		"@fortawesome/free-brands-svg-icons": "5.4.1", | ||||
| 		"@fortawesome/free-regular-svg-icons": "5.4.1", | ||||
| 		"@fortawesome/free-solid-svg-icons": "5.4.1", | ||||
| 		"@koa/cors": "2.2.2", | ||||
| 		"@prezzemolo/rap": "0.1.2", | ||||
| 		"@prezzemolo/zip": "0.0.3", | ||||
| 		"@types/bcryptjs": "2.4.2", | ||||
| 		"@types/chai-http": "3.0.5", | ||||
| 		"@types/dateformat": "1.0.1", | ||||
| 		"@types/debug": "0.0.31", | ||||
| 		"@types/deep-equal": "1.0.1", | ||||
| @@ -39,7 +40,7 @@ | ||||
| 		"@types/gulp-mocha": "0.0.32", | ||||
| 		"@types/gulp-rename": "0.0.33", | ||||
| 		"@types/gulp-replace": "0.0.31", | ||||
| 		"@types/gulp-uglify": "3.0.5", | ||||
| 		"@types/gulp-uglify": "3.0.6", | ||||
| 		"@types/gulp-util": "3.0.34", | ||||
| 		"@types/is-root": "1.0.0", | ||||
| 		"@types/is-url": "1.2.28", | ||||
| @@ -60,7 +61,7 @@ | ||||
| 		"@types/mocha": "5.2.3", | ||||
| 		"@types/mongodb": "3.1.12", | ||||
| 		"@types/ms": "0.7.30", | ||||
| 		"@types/node": "10.11.7", | ||||
| 		"@types/node": "10.12.0", | ||||
| 		"@types/portscanner": "2.1.0", | ||||
| 		"@types/pug": "2.0.4", | ||||
| 		"@types/qrcode": "1.3.0", | ||||
| @@ -70,7 +71,7 @@ | ||||
| 		"@types/request-promise-native": "1.0.15", | ||||
| 		"@types/rimraf": "2.0.2", | ||||
| 		"@types/seedrandom": "2.4.27", | ||||
| 		"@types/sharp": "0.17.10", | ||||
| 		"@types/sharp": "0.21.0", | ||||
| 		"@types/showdown": "1.7.5", | ||||
| 		"@types/single-line-log": "1.1.0", | ||||
| 		"@types/speakeasy": "2.0.2", | ||||
| @@ -78,7 +79,7 @@ | ||||
| 		"@types/tinycolor2": "1.4.1", | ||||
| 		"@types/tmp": "0.0.33", | ||||
| 		"@types/uuid": "3.4.4", | ||||
| 		"@types/webpack": "4.4.16", | ||||
| 		"@types/webpack": "4.4.17", | ||||
| 		"@types/webpack-stream": "3.2.10", | ||||
| 		"@types/websocket": "0.0.40", | ||||
| 		"@types/ws": "6.0.1", | ||||
| @@ -90,8 +91,10 @@ | ||||
| 		"bee-queue": "1.2.2", | ||||
| 		"bootstrap-vue": "2.0.0-rc.11", | ||||
| 		"cafy": "11.3.0", | ||||
| 		"chai": "4.2.0", | ||||
| 		"chai-http": "4.2.0", | ||||
| 		"chalk": "2.4.1", | ||||
| 		"chart.js": "2.7.2", | ||||
| 		"chart.js": "2.7.3", | ||||
| 		"commander": "2.19.0", | ||||
| 		"crc-32": "1.2.0", | ||||
| 		"css-loader": "1.0.0", | ||||
| @@ -157,7 +160,7 @@ | ||||
| 		"mkdirp": "0.5.1", | ||||
| 		"mocha": "5.2.0", | ||||
| 		"moji": "0.5.1", | ||||
| 		"mongodb": "3.1.1", | ||||
| 		"mongodb": "3.1.8", | ||||
| 		"monk": "6.0.6", | ||||
| 		"ms": "2.1.1", | ||||
| 		"nan": "2.11.1", | ||||
| @@ -220,7 +223,7 @@ | ||||
| 		"vue-loader": "15.4.2", | ||||
| 		"vue-router": "3.0.1", | ||||
| 		"vue-style-loader": "4.1.2", | ||||
| 		"vue-svg-inline-loader": "1.2.0", | ||||
| 		"vue-svg-inline-loader": "1.2.1", | ||||
| 		"vue-sweetalert2": "1.5.5", | ||||
| 		"vue-template-compiler": "2.5.17", | ||||
| 		"vuedraggable": "2.16.0", | ||||
|   | ||||
| @@ -44,7 +44,6 @@ export default define({ | ||||
| 		}, | ||||
| 		fetch() { | ||||
| 			fetch(`https://api.rss2json.com/v1/api.json?rss_url=${this.props.url}`, { | ||||
| 				cache: 'no-cache' | ||||
| 			}).then(res => { | ||||
| 				res.json().then(feed => { | ||||
| 					this.items = feed.items; | ||||
|   | ||||
| @@ -38,7 +38,7 @@ | ||||
| 			</div> | ||||
| 		</div> | ||||
| 	</div> | ||||
| 	<div class="main"> | ||||
| 	<div class="main" :class="{ side: widgets.left.length == 0 || widgets.right.length == 0 }"> | ||||
| 		<template v-if="customize"> | ||||
| 			<x-draggable v-for="place in ['left', 'right']" | ||||
| 				:list="widgets[place]" | ||||
| @@ -359,12 +359,10 @@ export default Vue.extend({ | ||||
| 				box-shadow var(--shadow) | ||||
| 				border-radius var(--round) | ||||
|  | ||||
| 			@media (max-width 700px) | ||||
| 				padding 0 | ||||
|  | ||||
| 				> .tl | ||||
| 					border none | ||||
| 					border-radius 0 | ||||
| 		&.side | ||||
| 			> .main | ||||
| 				width calc(100% - 280px) | ||||
| 				max-width 680px | ||||
|  | ||||
| 		> *:not(.main) | ||||
| 			width 280px | ||||
| @@ -381,14 +379,24 @@ export default Vue.extend({ | ||||
| 			padding-right 16px | ||||
| 			order 3 | ||||
|  | ||||
| 		@media (max-width 1100px) | ||||
| 			> *:not(.main) | ||||
| 				display none | ||||
| 		&.side | ||||
| 			@media (max-width 1000px) | ||||
| 				> *:not(.main) | ||||
| 					display none | ||||
|  | ||||
| 			> .main | ||||
| 				float none | ||||
| 				width 100% | ||||
| 				max-width 700px | ||||
| 				margin 0 auto | ||||
| 				> .main | ||||
| 					width 100% | ||||
| 					max-width 700px | ||||
| 					margin 0 auto | ||||
|  | ||||
| 		&:not(.side) | ||||
| 			@media (max-width 1200px) | ||||
| 				> *:not(.main) | ||||
| 					display none | ||||
|  | ||||
| 				> .main | ||||
| 					width 100% | ||||
| 					max-width 700px | ||||
| 					margin 0 auto | ||||
|  | ||||
| </style> | ||||
|   | ||||
| @@ -157,6 +157,9 @@ export default Vue.extend({ | ||||
| 			font-family Meiryo, sans-serif | ||||
| 			text-decoration none | ||||
|  | ||||
| 			@media (max-width 1100px) | ||||
| 				display none | ||||
|  | ||||
| 			[data-fa] | ||||
| 				margin-left 8px | ||||
|  | ||||
| @@ -171,6 +174,9 @@ export default Vue.extend({ | ||||
| 			border-radius 4px | ||||
| 			transition filter 100ms ease | ||||
|  | ||||
| 			@media (max-width 1100px) | ||||
| 				margin-left 8px | ||||
|  | ||||
| 	> .menu | ||||
| 		$bgcolor = var(--face) | ||||
| 		display block | ||||
|   | ||||
| @@ -29,6 +29,9 @@ export default Vue.extend({ | ||||
|  | ||||
| <style lang="stylus" scoped> | ||||
| .search | ||||
| 	@media (max-width 800px) | ||||
| 		display none !important | ||||
|  | ||||
| 	> [data-fa] | ||||
| 		display block | ||||
| 		position absolute | ||||
| @@ -58,6 +61,9 @@ export default Vue.extend({ | ||||
| 		transition color 0.5s ease, border 0.5s ease | ||||
| 		color var(--desktopHeaderSearchFg) | ||||
|  | ||||
| 		@media (max-width 1000px) | ||||
| 			width 10em | ||||
|  | ||||
| 		&::placeholder | ||||
| 			color var(--desktopHeaderFg) | ||||
|  | ||||
|   | ||||
| @@ -96,14 +96,10 @@ export default Vue.extend({ | ||||
| 		background-attachment fixed | ||||
| 		opacity 0.3 | ||||
|  | ||||
| 	> .header | ||||
| 		@media (max-width 1000px) | ||||
| 			display none | ||||
|  | ||||
| 	> .content.sidebar.left | ||||
| 		padding-left 64px | ||||
| 		padding-left 68px | ||||
|  | ||||
| 	> .content.sidebar.right | ||||
| 		padding-right 64px | ||||
| 		padding-right 68px | ||||
|  | ||||
| </style> | ||||
|   | ||||
| @@ -276,13 +276,24 @@ export default Vue.extend({ | ||||
| 	min-width 330px | ||||
| 	height 100% | ||||
| 	background var(--face) | ||||
| 	border-radius 6px | ||||
| 	//box-shadow 0 2px 16px rgba(#000, 0.1) | ||||
| 	border-radius var(--round) | ||||
| 	box-shadow var(--shadow) | ||||
| 	overflow hidden | ||||
|  | ||||
| 	&.draghover | ||||
| 		box-shadow 0 0 0 2px var(--primaryAlpha08) | ||||
|  | ||||
| 		&:after | ||||
| 			content "" | ||||
| 			display block | ||||
| 			position absolute | ||||
| 			z-index 1000 | ||||
| 			top 0 | ||||
| 			left 0 | ||||
| 			width 100% | ||||
| 			height 100% | ||||
| 			background var(--primaryAlpha02) | ||||
|  | ||||
| 	&.dragging | ||||
| 		box-shadow 0 0 0 2px var(--primaryAlpha04) | ||||
|  | ||||
| @@ -338,6 +349,7 @@ export default Vue.extend({ | ||||
|  | ||||
| 		> .toggleActive | ||||
| 		> .menu | ||||
| 			padding 0 | ||||
| 			width $header-height | ||||
| 			line-height $header-height | ||||
| 			font-size 16px | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| <template> | ||||
| 	<x-notes ref="timeline" :more="existMore ? more : null" :media-view="mediaView"/> | ||||
| <x-notes ref="timeline" :more="existMore ? more : null" :media-view="mediaView"/> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
|   | ||||
| @@ -15,7 +15,7 @@ | ||||
| 		<ui-switch v-model="column.isMediaView" @change="onChangeSettings">%i18n:@is-media-view%</ui-switch> | ||||
| 	</div> | ||||
| 	<x-list-tl v-if="column.type == 'list'" :list="column.list" :media-only="column.isMediaOnly" :media-view="column.isMediaView"/> | ||||
| 	<x-hashtag-tl v-if="column.type == 'hashtag'" :tag-tl="$store.state.settings.tagTimelines.find(x => x.id == column.tagTlId)" :media-only="column.isMediaOnly" :media-view="column.isMediaView"/> | ||||
| 	<x-hashtag-tl v-else-if="column.type == 'hashtag'" :tag-tl="$store.state.settings.tagTimelines.find(x => x.id == column.tagTlId)" :media-only="column.isMediaOnly" :media-view="column.isMediaView"/> | ||||
| 	<x-tl v-else :src="column.type" :media-only="column.isMediaOnly" :media-view="column.isMediaView"/> | ||||
| </x-column> | ||||
| </template> | ||||
|   | ||||
| @@ -41,6 +41,8 @@ | ||||
| 			<ui-button link :href="`${file.url}?download`" :download="file.name">%fa:download% %i18n:@download%</ui-button> | ||||
| 			<ui-button @click="rename">%fa:pencil-alt% %i18n:@rename%</ui-button> | ||||
| 			<ui-button @click="move">%fa:R folder-open% %i18n:@move%</ui-button> | ||||
| 			<ui-button @click="toggleSensitive" v-if="file.isSensitive">%fa:R eye% %i18n:@unmark-as-sensitive%</ui-button> | ||||
| 			<ui-button @click="toggleSensitive" v-else>%fa:R eye-slash% %i18n:@mark-as-sensitive%</ui-button> | ||||
| 			<ui-button @click="del">%fa:trash-alt R% %i18n:@delete%</ui-button> | ||||
| 		</div> | ||||
| 	</div> | ||||
| @@ -71,25 +73,30 @@ import { gcd } from '../../../../../prelude/math'; | ||||
|  | ||||
| export default Vue.extend({ | ||||
| 	props: ['file'], | ||||
|  | ||||
| 	data() { | ||||
| 		return { | ||||
| 			gcd, | ||||
| 			exif: null | ||||
| 		}; | ||||
| 	}, | ||||
|  | ||||
| 	computed: { | ||||
| 		browser(): any { | ||||
| 			return this.$parent; | ||||
| 		}, | ||||
|  | ||||
| 		kind(): string { | ||||
| 			return this.file.type.split('/')[0]; | ||||
| 		}, | ||||
|  | ||||
| 		style(): any { | ||||
| 			return this.file.properties.avgColor && this.file.properties.avgColor.length == 3 ? { | ||||
| 				'background-color': `rgb(${ this.file.properties.avgColor.join(',') })` | ||||
| 			} : {}; | ||||
| 		} | ||||
| 	}, | ||||
|  | ||||
| 	methods: { | ||||
| 		rename() { | ||||
| 			const name = window.prompt('%i18n:@rename%', this.file.name); | ||||
| @@ -101,6 +108,7 @@ export default Vue.extend({ | ||||
| 				this.browser.cf(this.file, true); | ||||
| 			}); | ||||
| 		}, | ||||
|  | ||||
| 		move() { | ||||
| 			(this as any).apis.chooseDriveFolder().then(folder => { | ||||
| 				(this as any).api('drive/files/update', { | ||||
| @@ -111,6 +119,7 @@ export default Vue.extend({ | ||||
| 				}); | ||||
| 			}); | ||||
| 		}, | ||||
|  | ||||
| 		del() { | ||||
| 			(this as any).api('drive/files/delete', { | ||||
| 				fileId: this.file.id | ||||
| @@ -118,9 +127,20 @@ export default Vue.extend({ | ||||
| 				this.browser.cd(this.file.folderId, true); | ||||
| 			}); | ||||
| 		}, | ||||
|  | ||||
| 		toggleSensitive() { | ||||
| 			(this as any).api('drive/files/update', { | ||||
| 				fileId: this.file.id, | ||||
| 				isSensitive: !this.file.isSensitive | ||||
| 			}); | ||||
|  | ||||
| 			this.file.isSensitive = !this.file.isSensitive; | ||||
| 		}, | ||||
|  | ||||
| 		showCreatedAt() { | ||||
| 			alert(new Date(this.file.createdAt).toLocaleString()); | ||||
| 		}, | ||||
|  | ||||
| 		onImageLoaded() { | ||||
| 			const self = this; | ||||
| 			EXIF.getData(this.$refs.img, function(this: any) { | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| import * as mongo from 'mongodb'; | ||||
| import { Context } from 'cafy'; | ||||
| import isObjectId from './is-objectid'; | ||||
|  | ||||
| export const isAnId = (x: any) => mongo.ObjectID.isValid(x); | ||||
| export const isNotAnId = (x: any) => !isAnId(x); | ||||
| @@ -12,7 +13,7 @@ export default class ID extends Context<mongo.ObjectID> { | ||||
| 		super(); | ||||
|  | ||||
| 		this.transform = v => { | ||||
| 			if (isAnId(v) && !mongo.ObjectID.prototype.isPrototypeOf(v)) { | ||||
| 			if (isAnId(v) && !isObjectId(v)) { | ||||
| 				return new mongo.ObjectID(v); | ||||
| 			} else { | ||||
| 				return v; | ||||
| @@ -20,7 +21,7 @@ export default class ID extends Context<mongo.ObjectID> { | ||||
| 		}; | ||||
|  | ||||
| 		this.push(v => { | ||||
| 			if (!mongo.ObjectID.prototype.isPrototypeOf(v) && isNotAnId(v)) { | ||||
| 			if (!isObjectId(v) && isNotAnId(v)) { | ||||
| 				return new Error('must-be-an-id'); | ||||
| 			} | ||||
| 			return true; | ||||
|   | ||||
							
								
								
									
										3
									
								
								src/misc/is-objectid.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								src/misc/is-objectid.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| export default function(x: any): boolean { | ||||
| 	return x.hasOwnProperty('toHexString') || x.hasOwnProperty('_bsontype'); | ||||
| } | ||||
| @@ -1,7 +1,8 @@ | ||||
| import * as mongo from 'mongodb'; | ||||
| import isObjectId from './is-objectid'; | ||||
|  | ||||
| function toString(id: any) { | ||||
| 	return mongo.ObjectID.prototype.isPrototypeOf(id) ? (id as mongo.ObjectID).toHexString() : id; | ||||
| 	return isObjectId(id) ? (id as mongo.ObjectID).toHexString() : id; | ||||
| } | ||||
|  | ||||
| export default function(note: any, mutedUserIds: string[]): boolean { | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| import * as mongo from 'mongodb'; | ||||
| import db from '../db/mongodb'; | ||||
| import isObjectId from '../misc/is-objectid'; | ||||
|  | ||||
| const AccessToken = db.get<IAccessToken>('accessTokens'); | ||||
| AccessToken.createIndex('token'); | ||||
| @@ -22,7 +23,7 @@ export async function deleteAccessToken(accessToken: string | mongo.ObjectID | I | ||||
| 	let a: IAccessToken; | ||||
|  | ||||
| 	// Populate | ||||
| 	if (mongo.ObjectID.prototype.isPrototypeOf(accessToken)) { | ||||
| 	if (isObjectId(accessToken)) { | ||||
| 		a = await AccessToken.findOne({ | ||||
| 			_id: accessToken | ||||
| 		}); | ||||
|   | ||||
| @@ -2,6 +2,7 @@ import * as mongo from 'mongodb'; | ||||
| const deepcopy = require('deepcopy'); | ||||
| import AccessToken from './access-token'; | ||||
| import db from '../db/mongodb'; | ||||
| import isObjectId from '../misc/is-objectid'; | ||||
| import config from '../config'; | ||||
|  | ||||
| const App = db.get<IApp>('apps'); | ||||
| @@ -43,7 +44,7 @@ export const pack = ( | ||||
| 	let _app: any; | ||||
|  | ||||
| 	// Populate the app if 'app' is ID | ||||
| 	if (mongo.ObjectID.prototype.isPrototypeOf(app)) { | ||||
| 	if (isObjectId(app)) { | ||||
| 		_app = await App.findOne({ | ||||
| 			_id: app | ||||
| 		}); | ||||
| @@ -56,7 +57,7 @@ export const pack = ( | ||||
| 	} | ||||
|  | ||||
| 	// Me | ||||
| 	if (me && !mongo.ObjectID.prototype.isPrototypeOf(me)) { | ||||
| 	if (me && !isObjectId(me)) { | ||||
| 		if (typeof me === 'string') { | ||||
| 			me = new mongo.ObjectID(me); | ||||
| 		} else { | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| import * as mongo from 'mongodb'; | ||||
| const deepcopy = require('deepcopy'); | ||||
| import db from '../db/mongodb'; | ||||
| import isObjectId from '../misc/is-objectid'; | ||||
| import { pack as packApp } from './app'; | ||||
|  | ||||
| const AuthSession = db.get<IAuthSession>('authSessions'); | ||||
| @@ -31,7 +32,7 @@ export const pack = ( | ||||
| 	_session = deepcopy(session); | ||||
|  | ||||
| 	// Me | ||||
| 	if (me && !mongo.ObjectID.prototype.isPrototypeOf(me)) { | ||||
| 	if (me && !isObjectId(me)) { | ||||
| 		if (typeof me === 'string') { | ||||
| 			me = new mongo.ObjectID(me); | ||||
| 		} else { | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| import * as mongo from 'mongodb'; | ||||
| import monkDb, { nativeDbConn } from '../db/mongodb'; | ||||
| import isObjectId from '../misc/is-objectid'; | ||||
|  | ||||
| const DriveFileThumbnail = monkDb.get<IDriveFileThumbnail>('driveFileThumbnails.files'); | ||||
| DriveFileThumbnail.createIndex('metadata.originalId', { sparse: true, unique: true }); | ||||
| @@ -35,7 +36,7 @@ export async function deleteDriveFileThumbnail(driveFile: string | mongo.ObjectI | ||||
| 	let d: IDriveFileThumbnail; | ||||
|  | ||||
| 	// Populate | ||||
| 	if (mongo.ObjectID.prototype.isPrototypeOf(driveFile)) { | ||||
| 	if (isObjectId(driveFile)) { | ||||
| 		d = await DriveFileThumbnail.findOne({ | ||||
| 			_id: driveFile | ||||
| 		}); | ||||
|   | ||||
| @@ -3,6 +3,7 @@ const deepcopy = require('deepcopy'); | ||||
| import { pack as packFolder } from './drive-folder'; | ||||
| import config from '../config'; | ||||
| import monkDb, { nativeDbConn } from '../db/mongodb'; | ||||
| import isObjectId from '../misc/is-objectid'; | ||||
| import Note, { deleteNote } from './note'; | ||||
| import MessagingMessage, { deleteMessagingMessage } from './messaging-message'; | ||||
| import User from './user'; | ||||
| @@ -78,7 +79,7 @@ export async function deleteDriveFile(driveFile: string | mongo.ObjectID | IDriv | ||||
| 	let d: IDriveFile; | ||||
|  | ||||
| 	// Populate | ||||
| 	if (mongo.ObjectID.prototype.isPrototypeOf(driveFile)) { | ||||
| 	if (isObjectId(driveFile)) { | ||||
| 		d = await DriveFile.findOne({ | ||||
| 			_id: driveFile | ||||
| 		}); | ||||
| @@ -154,7 +155,7 @@ export const pack = ( | ||||
| 	let _file: any; | ||||
|  | ||||
| 	// Populate the file if 'file' is ID | ||||
| 	if (mongo.ObjectID.prototype.isPrototypeOf(file)) { | ||||
| 	if (isObjectId(file)) { | ||||
| 		_file = await DriveFile.findOne({ | ||||
| 			_id: file | ||||
| 		}); | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| import * as mongo from 'mongodb'; | ||||
| const deepcopy = require('deepcopy'); | ||||
| import db from '../db/mongodb'; | ||||
| import isObjectId from '../misc/is-objectid'; | ||||
| import DriveFile from './drive-file'; | ||||
|  | ||||
| const DriveFolder = db.get<IDriveFolder>('driveFolders'); | ||||
| @@ -29,7 +30,7 @@ export async function deleteDriveFolder(driveFolder: string | mongo.ObjectID | I | ||||
| 	let d: IDriveFolder; | ||||
|  | ||||
| 	// Populate | ||||
| 	if (mongo.ObjectID.prototype.isPrototypeOf(driveFolder)) { | ||||
| 	if (isObjectId(driveFolder)) { | ||||
| 		d = await DriveFolder.findOne({ | ||||
| 			_id: driveFolder | ||||
| 		}); | ||||
| @@ -83,7 +84,7 @@ export const pack = ( | ||||
| 	let _folder: any; | ||||
|  | ||||
| 	// Populate the folder if 'folder' is ID | ||||
| 	if (mongo.ObjectID.prototype.isPrototypeOf(folder)) { | ||||
| 	if (isObjectId(folder)) { | ||||
| 		_folder = await DriveFolder.findOne({ _id: folder }); | ||||
| 	} else if (typeof folder === 'string') { | ||||
| 		_folder = await DriveFolder.findOne({ _id: new mongo.ObjectID(folder) }); | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| import * as mongo from 'mongodb'; | ||||
| const deepcopy = require('deepcopy'); | ||||
| import db from '../db/mongodb'; | ||||
| import isObjectId from '../misc/is-objectid'; | ||||
| import { pack as packNote } from './note'; | ||||
|  | ||||
| const Favorite = db.get<IFavorite>('favorites'); | ||||
| @@ -21,7 +22,7 @@ export async function deleteFavorite(favorite: string | mongo.ObjectID | IFavori | ||||
| 	let f: IFavorite; | ||||
|  | ||||
| 	// Populate | ||||
| 	if (mongo.ObjectID.prototype.isPrototypeOf(favorite)) { | ||||
| 	if (isObjectId(favorite)) { | ||||
| 		f = await Favorite.findOne({ | ||||
| 			_id: favorite | ||||
| 		}); | ||||
| @@ -58,7 +59,7 @@ export const pack = ( | ||||
| 	let _favorite: any; | ||||
|  | ||||
| 	// Populate the favorite if 'favorite' is ID | ||||
| 	if (mongo.ObjectID.prototype.isPrototypeOf(favorite)) { | ||||
| 	if (isObjectId(favorite)) { | ||||
| 		_favorite = await Favorite.findOne({ | ||||
| 			_id: favorite | ||||
| 		}); | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| import * as mongo from 'mongodb'; | ||||
| const deepcopy = require('deepcopy'); | ||||
| import db from '../db/mongodb'; | ||||
| import isObjectId from '../misc/is-objectid'; | ||||
| import { pack as packUser } from './user'; | ||||
|  | ||||
| const FollowRequest = db.get<IFollowRequest>('followRequests'); | ||||
| @@ -12,6 +13,7 @@ export type IFollowRequest = { | ||||
| 	createdAt: Date; | ||||
| 	followeeId: mongo.ObjectID; | ||||
| 	followerId: mongo.ObjectID; | ||||
| 	requestId?: string;	// id of Follow Activity | ||||
|  | ||||
| 	// 非正規化 | ||||
| 	_followee: { | ||||
| @@ -33,7 +35,7 @@ export async function deleteFollowRequest(followRequest: string | mongo.ObjectID | ||||
| 	let f: IFollowRequest; | ||||
|  | ||||
| 	// Populate | ||||
| 	if (mongo.ObjectID.prototype.isPrototypeOf(followRequest)) { | ||||
| 	if (isObjectId(followRequest)) { | ||||
| 		f = await FollowRequest.findOne({ | ||||
| 			_id: followRequest | ||||
| 		}); | ||||
| @@ -63,7 +65,7 @@ export const pack = ( | ||||
| 	let _request: any; | ||||
|  | ||||
| 	// Populate the request if 'request' is ID | ||||
| 	if (mongo.ObjectID.prototype.isPrototypeOf(request)) { | ||||
| 	if (isObjectId(request)) { | ||||
| 		_request = await FollowRequest.findOne({ | ||||
| 			_id: request | ||||
| 		}); | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| import * as mongo from 'mongodb'; | ||||
| import db from '../db/mongodb'; | ||||
| import isObjectId from '../misc/is-objectid'; | ||||
|  | ||||
| const FollowedLog = db.get<IFollowedLog>('followedLogs'); | ||||
| export default FollowedLog; | ||||
| @@ -18,7 +19,7 @@ export async function deleteFollowedLog(followedLog: string | mongo.ObjectID | I | ||||
| 	let f: IFollowedLog; | ||||
|  | ||||
| 	// Populate | ||||
| 	if (mongo.ObjectID.prototype.isPrototypeOf(followedLog)) { | ||||
| 	if (isObjectId(followedLog)) { | ||||
| 		f = await FollowedLog.findOne({ | ||||
| 			_id: followedLog | ||||
| 		}); | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| import * as mongo from 'mongodb'; | ||||
| import db from '../db/mongodb'; | ||||
| import isObjectId from '../misc/is-objectid'; | ||||
|  | ||||
| const FollowingLog = db.get<IFollowingLog>('followingLogs'); | ||||
| export default FollowingLog; | ||||
| @@ -18,7 +19,7 @@ export async function deleteFollowingLog(followingLog: string | mongo.ObjectID | | ||||
| 	let f: IFollowingLog; | ||||
|  | ||||
| 	// Populate | ||||
| 	if (mongo.ObjectID.prototype.isPrototypeOf(followingLog)) { | ||||
| 	if (isObjectId(followingLog)) { | ||||
| 		f = await FollowingLog.findOne({ | ||||
| 			_id: followingLog | ||||
| 		}); | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| import * as mongo from 'mongodb'; | ||||
| import db from '../db/mongodb'; | ||||
| import isObjectId from '../misc/is-objectid'; | ||||
|  | ||||
| const Following = db.get<IFollowing>('following'); | ||||
| Following.createIndex(['followerId', 'followeeId'], { unique: true }); | ||||
| @@ -32,7 +33,7 @@ export async function deleteFollowing(following: string | mongo.ObjectID | IFoll | ||||
| 	let f: IFollowing; | ||||
|  | ||||
| 	// Populate | ||||
| 	if (mongo.ObjectID.prototype.isPrototypeOf(following)) { | ||||
| 	if (isObjectId(following)) { | ||||
| 		f = await Following.findOne({ | ||||
| 			_id: following | ||||
| 		}); | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| import * as mongo from 'mongodb'; | ||||
| const deepcopy = require('deepcopy'); | ||||
| import db from '../../../db/mongodb'; | ||||
| import isObjectId from '../../../misc/is-objectid'; | ||||
| import { IUser, pack as packUser } from '../../user'; | ||||
|  | ||||
| const ReversiGame = db.get<IReversiGame>('reversiGames'); | ||||
| @@ -62,7 +63,7 @@ export const pack = ( | ||||
| 	let _game: any; | ||||
|  | ||||
| 	// Populate the game if 'game' is ID | ||||
| 	if (mongo.ObjectID.prototype.isPrototypeOf(game)) { | ||||
| 	if (isObjectId(game)) { | ||||
| 		_game = await ReversiGame.findOne({ | ||||
| 			_id: game | ||||
| 		}); | ||||
| @@ -76,7 +77,7 @@ export const pack = ( | ||||
|  | ||||
| 	// Me | ||||
| 	const meId: mongo.ObjectID = me | ||||
| 		? mongo.ObjectID.prototype.isPrototypeOf(me) | ||||
| 		? isObjectId(me) | ||||
| 			? me as mongo.ObjectID | ||||
| 			: typeof me === 'string' | ||||
| 				? new mongo.ObjectID(me) | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| import * as mongo from 'mongodb'; | ||||
| const deepcopy = require('deepcopy'); | ||||
| import db from '../../../db/mongodb'; | ||||
| import isObjectId from '../../../misc/is-objectid'; | ||||
| import { IUser, pack as packUser } from '../../user'; | ||||
|  | ||||
| const Matching = db.get<IMatching>('reversiMatchings'); | ||||
| @@ -23,7 +24,7 @@ export const pack = ( | ||||
|  | ||||
| 	// Me | ||||
| 	const meId: mongo.ObjectID = me | ||||
| 		? mongo.ObjectID.prototype.isPrototypeOf(me) | ||||
| 		? isObjectId(me) | ||||
| 			? me as mongo.ObjectID | ||||
| 			: typeof me === 'string' | ||||
| 				? new mongo.ObjectID(me) | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| import * as mongo from 'mongodb'; | ||||
| import db from '../db/mongodb'; | ||||
| import isObjectId from '../misc/is-objectid'; | ||||
|  | ||||
| const MessagingHistory = db.get<IMessagingHistory>('messagingHistories'); | ||||
| export default MessagingHistory; | ||||
| @@ -19,7 +20,7 @@ export async function deleteMessagingHistory(messagingHistory: string | mongo.Ob | ||||
| 	let m: IMessagingHistory; | ||||
|  | ||||
| 	// Populate | ||||
| 	if (mongo.ObjectID.prototype.isPrototypeOf(messagingHistory)) { | ||||
| 	if (isObjectId(messagingHistory)) { | ||||
| 		m = await MessagingHistory.findOne({ | ||||
| 			_id: messagingHistory | ||||
| 		}); | ||||
|   | ||||
| @@ -3,6 +3,7 @@ const deepcopy = require('deepcopy'); | ||||
| import { pack as packUser } from './user'; | ||||
| import { pack as packFile } from './drive-file'; | ||||
| import db from '../db/mongodb'; | ||||
| import isObjectId from '../misc/is-objectid'; | ||||
| import MessagingHistory, { deleteMessagingHistory } from './messaging-history'; | ||||
| import { length } from 'stringz'; | ||||
|  | ||||
| @@ -30,7 +31,7 @@ export async function deleteMessagingMessage(messagingMessage: string | mongo.Ob | ||||
| 	let m: IMessagingMessage; | ||||
|  | ||||
| 	// Populate | ||||
| 	if (mongo.ObjectID.prototype.isPrototypeOf(messagingMessage)) { | ||||
| 	if (isObjectId(messagingMessage)) { | ||||
| 		m = await MessagingMessage.findOne({ | ||||
| 			_id: messagingMessage | ||||
| 		}); | ||||
| @@ -72,7 +73,7 @@ export const pack = ( | ||||
| 	let _message: any; | ||||
|  | ||||
| 	// Populate the message if 'message' is ID | ||||
| 	if (mongo.ObjectID.prototype.isPrototypeOf(message)) { | ||||
| 	if (isObjectId(message)) { | ||||
| 		_message = await MessagingMessage.findOne({ | ||||
| 			_id: message | ||||
| 		}); | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| import * as mongo from 'mongodb'; | ||||
| import db from '../db/mongodb'; | ||||
| import isObjectId from '../misc/is-objectid'; | ||||
|  | ||||
| const Mute = db.get<IMute>('mute'); | ||||
| Mute.createIndex(['muterId', 'muteeId'], { unique: true }); | ||||
| @@ -19,7 +20,7 @@ export async function deleteMute(mute: string | mongo.ObjectID | IMute) { | ||||
| 	let m: IMute; | ||||
|  | ||||
| 	// Populate | ||||
| 	if (mongo.ObjectID.prototype.isPrototypeOf(mute)) { | ||||
| 	if (isObjectId(mute)) { | ||||
| 		m = await Mute.findOne({ | ||||
| 			_id: mute | ||||
| 		}); | ||||
|   | ||||
| @@ -2,6 +2,7 @@ import * as mongo from 'mongodb'; | ||||
| import $ from 'cafy'; | ||||
| const deepcopy = require('deepcopy'); | ||||
| import db from '../db/mongodb'; | ||||
| import isObjectId from '../misc/is-objectid'; | ||||
| import Reaction from './note-reaction'; | ||||
| import { pack as packUser } from './user'; | ||||
|  | ||||
| @@ -37,7 +38,7 @@ export async function deleteNoteReaction(noteReaction: string | mongo.ObjectID | | ||||
| 	let n: INoteReaction; | ||||
|  | ||||
| 	// Populate | ||||
| 	if (mongo.ObjectID.prototype.isPrototypeOf(noteReaction)) { | ||||
| 	if (isObjectId(noteReaction)) { | ||||
| 		n = await NoteReaction.findOne({ | ||||
| 			_id: noteReaction | ||||
| 		}); | ||||
| @@ -67,7 +68,7 @@ export const pack = ( | ||||
| 	let _reaction: any; | ||||
|  | ||||
| 	// Populate the reaction if 'reaction' is ID | ||||
| 	if (mongo.ObjectID.prototype.isPrototypeOf(reaction)) { | ||||
| 	if (isObjectId(reaction)) { | ||||
| 		_reaction = await Reaction.findOne({ | ||||
| 			_id: reaction | ||||
| 		}); | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| import * as mongo from 'mongodb'; | ||||
| import db from '../db/mongodb'; | ||||
| import isObjectId from '../misc/is-objectid'; | ||||
|  | ||||
| const NoteWatching = db.get<INoteWatching>('noteWatching'); | ||||
| NoteWatching.createIndex(['userId', 'noteId'], { unique: true }); | ||||
| @@ -19,7 +20,7 @@ export async function deleteNoteWatching(noteWatching: string | mongo.ObjectID | | ||||
| 	let n: INoteWatching; | ||||
|  | ||||
| 	// Populate | ||||
| 	if (mongo.ObjectID.prototype.isPrototypeOf(noteWatching)) { | ||||
| 	if (isObjectId(noteWatching)) { | ||||
| 		n = await NoteWatching.findOne({ | ||||
| 			_id: noteWatching | ||||
| 		}); | ||||
|   | ||||
| @@ -2,6 +2,7 @@ import * as mongo from 'mongodb'; | ||||
| const deepcopy = require('deepcopy'); | ||||
| import rap from '@prezzemolo/rap'; | ||||
| import db from '../db/mongodb'; | ||||
| import isObjectId from '../misc/is-objectid'; | ||||
| import { length } from 'stringz'; | ||||
| import { IUser, pack as packUser } from './user'; | ||||
| import { pack as packApp } from './app'; | ||||
| @@ -107,7 +108,7 @@ export async function deleteNote(note: string | mongo.ObjectID | INote) { | ||||
| 	let n: INote; | ||||
|  | ||||
| 	// Populate | ||||
| 	if (mongo.ObjectID.prototype.isPrototypeOf(note)) { | ||||
| 	if (isObjectId(note)) { | ||||
| 		n = await Note.findOne({ | ||||
| 			_id: note | ||||
| 		}); | ||||
| @@ -259,7 +260,7 @@ export const pack = async ( | ||||
|  | ||||
| 	// Me | ||||
| 	const meId: mongo.ObjectID = me | ||||
| 		? mongo.ObjectID.prototype.isPrototypeOf(me) | ||||
| 		? isObjectId(me) | ||||
| 			? me as mongo.ObjectID | ||||
| 			: typeof me === 'string' | ||||
| 				? new mongo.ObjectID(me) | ||||
| @@ -269,7 +270,7 @@ export const pack = async ( | ||||
| 	let _note: any; | ||||
|  | ||||
| 	// Populate the note if 'note' is ID | ||||
| 	if (mongo.ObjectID.prototype.isPrototypeOf(note)) { | ||||
| 	if (isObjectId(note)) { | ||||
| 		_note = await Note.findOne({ | ||||
| 			_id: note | ||||
| 		}); | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| import * as mongo from 'mongodb'; | ||||
| const deepcopy = require('deepcopy'); | ||||
| import db from '../db/mongodb'; | ||||
| import isObjectId from '../misc/is-objectid'; | ||||
| import { IUser, pack as packUser } from './user'; | ||||
| import { pack as packNote } from './note'; | ||||
|  | ||||
| @@ -57,7 +58,7 @@ export async function deleteNotification(notification: string | mongo.ObjectID | | ||||
| 	let n: INotification; | ||||
|  | ||||
| 	// Populate | ||||
| 	if (mongo.ObjectID.prototype.isPrototypeOf(notification)) { | ||||
| 	if (isObjectId(notification)) { | ||||
| 		n = await Notification.findOne({ | ||||
| 			_id: notification | ||||
| 		}); | ||||
| @@ -90,7 +91,7 @@ export const pack = (notification: any) => new Promise<any>(async (resolve, reje | ||||
| 	let _notification: any; | ||||
|  | ||||
| 	// Populate the notification if 'notification' is ID | ||||
| 	if (mongo.ObjectID.prototype.isPrototypeOf(notification)) { | ||||
| 	if (isObjectId(notification)) { | ||||
| 		_notification = await Notification.findOne({ | ||||
| 			_id: notification | ||||
| 		}); | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| import * as mongo from 'mongodb'; | ||||
| import db from '../db/mongodb'; | ||||
| import isObjectId from '../misc/is-objectid'; | ||||
|  | ||||
| const PollVote = db.get<IPollVote>('pollVotes'); | ||||
| export default PollVote; | ||||
| @@ -19,7 +20,7 @@ export async function deletePollVote(pollVote: string | mongo.ObjectID | IPollVo | ||||
| 	let p: IPollVote; | ||||
|  | ||||
| 	// Populate | ||||
| 	if (mongo.ObjectID.prototype.isPrototypeOf(pollVote)) { | ||||
| 	if (isObjectId(pollVote)) { | ||||
| 		p = await PollVote.findOne({ | ||||
| 			_id: pollVote | ||||
| 		}); | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| import * as mongo from 'mongodb'; | ||||
| import db from '../db/mongodb'; | ||||
| import isObjectId from '../misc/is-objectid'; | ||||
|  | ||||
| const SwSubscription = db.get<ISwSubscription>('swSubscriptions'); | ||||
| export default SwSubscription; | ||||
| @@ -19,7 +20,7 @@ export async function deleteSwSubscription(swSubscription: string | mongo.Object | ||||
| 	let s: ISwSubscription; | ||||
|  | ||||
| 	// Populate | ||||
| 	if (mongo.ObjectID.prototype.isPrototypeOf(swSubscription)) { | ||||
| 	if (isObjectId(swSubscription)) { | ||||
| 		s = await SwSubscription.findOne({ | ||||
| 			_id: swSubscription | ||||
| 		}); | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| import * as mongo from 'mongodb'; | ||||
| const deepcopy = require('deepcopy'); | ||||
| import db from '../db/mongodb'; | ||||
| import isObjectId from '../misc/is-objectid'; | ||||
|  | ||||
| const UserList = db.get<IUserList>('userList'); | ||||
| export default UserList; | ||||
| @@ -20,7 +21,7 @@ export async function deleteUserList(userList: string | mongo.ObjectID | IUserLi | ||||
| 	let u: IUserList; | ||||
|  | ||||
| 	// Populate | ||||
| 	if (mongo.ObjectID.prototype.isPrototypeOf(userList)) { | ||||
| 	if (isObjectId(userList)) { | ||||
| 		u = await UserList.findOne({ | ||||
| 			_id: userList | ||||
| 		}); | ||||
| @@ -45,7 +46,7 @@ export const pack = ( | ||||
| ) => new Promise<any>(async (resolve, reject) => { | ||||
| 	let _userList: any; | ||||
|  | ||||
| 	if (mongo.ObjectID.prototype.isPrototypeOf(userList)) { | ||||
| 	if (isObjectId(userList)) { | ||||
| 		_userList = await UserList.findOne({ | ||||
| 			_id: userList | ||||
| 		}); | ||||
|   | ||||
| @@ -3,6 +3,7 @@ const deepcopy = require('deepcopy'); | ||||
| const sequential = require('promise-sequential'); | ||||
| import rap from '@prezzemolo/rap'; | ||||
| import db from '../db/mongodb'; | ||||
| import isObjectId from '../misc/is-objectid'; | ||||
| import Note, { packMany as packNoteMany, deleteNote } from './note'; | ||||
| import Following, { deleteFollowing } from './following'; | ||||
| import Mute, { deleteMute } from './mute'; | ||||
| @@ -175,7 +176,7 @@ export async function deleteUser(user: string | mongo.ObjectID | IUser) { | ||||
| 	let u: IUser; | ||||
|  | ||||
| 	// Populate | ||||
| 	if (mongo.ObjectID.prototype.isPrototypeOf(user)) { | ||||
| 	if (isObjectId(user)) { | ||||
| 		u = await User.findOne({ | ||||
| 			_id: user | ||||
| 		}); | ||||
| @@ -340,7 +341,6 @@ export const pack = ( | ||||
| 		includeHasUnreadNotes?: boolean | ||||
| 	} | ||||
| ) => new Promise<any>(async (resolve, reject) => { | ||||
|  | ||||
| 	const opts = Object.assign({ | ||||
| 		detail: false, | ||||
| 		includeSecrets: false | ||||
| @@ -358,7 +358,7 @@ export const pack = ( | ||||
| 	}; | ||||
|  | ||||
| 	// Populate the user if 'user' is ID | ||||
| 	if (mongo.ObjectID.prototype.isPrototypeOf(user)) { | ||||
| 	if (isObjectId(user)) { | ||||
| 		_user = await User.findOne({ | ||||
| 			_id: user | ||||
| 		}, { fields }); | ||||
| @@ -378,7 +378,7 @@ export const pack = ( | ||||
|  | ||||
| 	// Me | ||||
| 	const meId: mongo.ObjectID = me | ||||
| 		? mongo.ObjectID.prototype.isPrototypeOf(me) | ||||
| 		? isObjectId(me) | ||||
| 			? me as mongo.ObjectID | ||||
| 			: typeof me === 'string' | ||||
| 				? new mongo.ObjectID(me) | ||||
|   | ||||
| @@ -23,5 +23,5 @@ export default async (actor: IRemoteUser, activity: IFollow): Promise<void> => { | ||||
| 		throw new Error('フォローしようとしているユーザーはローカルユーザーではありません'); | ||||
| 	} | ||||
|  | ||||
| 	await follow(actor, followee); | ||||
| 	await follow(actor, followee, activity.id); | ||||
| }; | ||||
|   | ||||
| @@ -1,4 +1,8 @@ | ||||
| export default (object: any) => ({ | ||||
| import config from '../../../config'; | ||||
| import { ILocalUser } from '../../../models/user'; | ||||
|  | ||||
| export default (object: any, user: ILocalUser) => ({ | ||||
| 	type: 'Accept', | ||||
| 	actor: `${config.url}/users/${user._id}`, | ||||
| 	object | ||||
| }); | ||||
|   | ||||
| @@ -1,8 +1,14 @@ | ||||
| import config from '../../../config'; | ||||
| import { IUser, isLocalUser } from '../../../models/user'; | ||||
|  | ||||
| export default (follower: IUser, followee: IUser) => ({ | ||||
| 	type: 'Follow', | ||||
| 	actor: isLocalUser(follower) ? `${config.url}/users/${follower._id}` : follower.uri, | ||||
| 	object: isLocalUser(followee) ? `${config.url}/users/${followee._id}` : followee.uri | ||||
| }); | ||||
| export default (follower: IUser, followee: IUser, requestId?: string) => { | ||||
| 	const follow = { | ||||
| 		type: 'Follow', | ||||
| 		actor: isLocalUser(follower) ? `${config.url}/users/${follower._id}` : follower.uri, | ||||
| 		object: isLocalUser(followee) ? `${config.url}/users/${followee._id}` : followee.uri | ||||
| 	} as any; | ||||
|  | ||||
| 	if (requestId) follow.id = requestId; | ||||
|  | ||||
| 	return follow; | ||||
| }; | ||||
|   | ||||
| @@ -1,4 +1,8 @@ | ||||
| export default (object: any) => ({ | ||||
| import config from '../../../config'; | ||||
| import { ILocalUser } from '../../../models/user'; | ||||
|  | ||||
| export default (object: any, user: ILocalUser) => ({ | ||||
| 	type: 'Reject', | ||||
| 	actor: `${config.url}/users/${user._id}`, | ||||
| 	object | ||||
| }); | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| import * as mongo from 'mongodb'; | ||||
| import isObjectId from '../../../misc/is-objectid'; | ||||
| import Message from '../../../models/messaging-message'; | ||||
| import { IMessagingMessage as IMessage } from '../../../models/messaging-message'; | ||||
| import { publishMainStream } from '../../../stream'; | ||||
| @@ -15,21 +16,21 @@ export default ( | ||||
| 	message: string | string[] | IMessage | IMessage[] | mongo.ObjectID | mongo.ObjectID[] | ||||
| ) => new Promise<any>(async (resolve, reject) => { | ||||
|  | ||||
| 	const userId = mongo.ObjectID.prototype.isPrototypeOf(user) | ||||
| 	const userId = isObjectId(user) | ||||
| 		? user | ||||
| 		: new mongo.ObjectID(user); | ||||
|  | ||||
| 	const otherpartyId = mongo.ObjectID.prototype.isPrototypeOf(otherparty) | ||||
| 	const otherpartyId = isObjectId(otherparty) | ||||
| 		? otherparty | ||||
| 		: new mongo.ObjectID(otherparty); | ||||
|  | ||||
| 	const ids: mongo.ObjectID[] = Array.isArray(message) | ||||
| 		? mongo.ObjectID.prototype.isPrototypeOf(message[0]) | ||||
| 		? isObjectId(message[0]) | ||||
| 			? (message as mongo.ObjectID[]) | ||||
| 			: typeof message[0] === 'string' | ||||
| 				? (message as string[]).map(m => new mongo.ObjectID(m)) | ||||
| 				: (message as IMessage[]).map(m => m._id) | ||||
| 		: mongo.ObjectID.prototype.isPrototypeOf(message) | ||||
| 		: isObjectId(message) | ||||
| 			? [(message as mongo.ObjectID)] | ||||
| 			: typeof message === 'string' | ||||
| 				? [new mongo.ObjectID(message)] | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| import * as mongo from 'mongodb'; | ||||
| import isObjectId from '../../../misc/is-objectid'; | ||||
| import { default as Notification, INotification } from '../../../models/notification'; | ||||
| import { publishMainStream } from '../../../stream'; | ||||
| import Mute from '../../../models/mute'; | ||||
| @@ -12,17 +13,17 @@ export default ( | ||||
| 	message: string | string[] | INotification | INotification[] | mongo.ObjectID | mongo.ObjectID[] | ||||
| ) => new Promise<any>(async (resolve, reject) => { | ||||
|  | ||||
| 	const userId = mongo.ObjectID.prototype.isPrototypeOf(user) | ||||
| 	const userId = isObjectId(user) | ||||
| 		? user | ||||
| 		: new mongo.ObjectID(user); | ||||
|  | ||||
| 	const ids: mongo.ObjectID[] = Array.isArray(message) | ||||
| 		? mongo.ObjectID.prototype.isPrototypeOf(message[0]) | ||||
| 		? isObjectId(message[0]) | ||||
| 			? (message as mongo.ObjectID[]) | ||||
| 			: typeof message[0] === 'string' | ||||
| 				? (message as string[]).map(m => new mongo.ObjectID(m)) | ||||
| 				: (message as INotification[]).map(m => m._id) | ||||
| 		: mongo.ObjectID.prototype.isPrototypeOf(message) | ||||
| 		: isObjectId(message) | ||||
| 			? [(message as mongo.ObjectID)] | ||||
| 			: typeof message === 'string' | ||||
| 				? [new mongo.ObjectID(message)] | ||||
|   | ||||
| @@ -101,7 +101,7 @@ export const meta = { | ||||
|  | ||||
| export default async (params: any, user: ILocalUser, app: IApp) => new Promise(async (res, rej) => { | ||||
| 	const [ps, psErr] = getParams(meta, params); | ||||
| 	if (psErr) throw psErr; | ||||
| 	if (psErr) return rej(psErr); | ||||
|  | ||||
| 	const isSecure = user != null && app == null; | ||||
|  | ||||
|   | ||||
| @@ -46,6 +46,8 @@ router.post('/signin', require('./private/signin').default); | ||||
| router.use(require('./service/github').routes()); | ||||
| router.use(require('./service/twitter').routes()); | ||||
|  | ||||
| router.use(require('./mastodon').routes()); | ||||
|  | ||||
| // Return 404 for unknown API | ||||
| router.all('*', async ctx => { | ||||
| 	ctx.status = 404; | ||||
| @@ -54,4 +56,4 @@ router.all('*', async ctx => { | ||||
| // Register router | ||||
| app.use(router.routes()); | ||||
|  | ||||
| module.exports = app; | ||||
| export default app; | ||||
|   | ||||
							
								
								
									
										14
									
								
								src/server/api/mastodon.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								src/server/api/mastodon.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| import * as Router from 'koa-router'; | ||||
| import User from '../../models/user'; | ||||
| import { toASCII } from 'punycode'; | ||||
|  | ||||
| // Init router | ||||
| const router = new Router(); | ||||
|  | ||||
| router.get('/v1/instance/peers', async ctx => { | ||||
| 	const peers = await User.distinct('host', { host: { $ne: null } }) as any as string[]; | ||||
| 	const punyCodes = peers.map(peer => toASCII(peer)); | ||||
| 	ctx.body = punyCodes; | ||||
| }); | ||||
|  | ||||
| module.exports = router; | ||||
| @@ -12,9 +12,8 @@ export default async (ctx: Koa.Context) => { | ||||
| 	ctx.set('Access-Control-Allow-Credentials', 'true'); | ||||
|  | ||||
| 	const body = ctx.request.body as any; | ||||
| 	// See: https://github.com/syuilo/misskey/issues/2384 | ||||
| 	const username = body['username'] || body['x']; | ||||
| 	const password = body['password'] || body['y']; | ||||
| 	const username = body['username']; | ||||
| 	const password = body['password']; | ||||
| 	const token = body['token']; | ||||
|  | ||||
| 	if (typeof username != 'string') { | ||||
|   | ||||
| @@ -132,6 +132,12 @@ export default async (ctx: Koa.Context) => { | ||||
|  | ||||
| 	updateUserStats(account, true); | ||||
|  | ||||
| 	// Response | ||||
| 	ctx.body = await pack(account); | ||||
| 	const res = await pack(account, account, { | ||||
| 		detail: true, | ||||
| 		includeSecrets: true | ||||
| 	}); | ||||
|  | ||||
| 	res.token = secret; | ||||
|  | ||||
| 	ctx.body = res; | ||||
| }; | ||||
|   | ||||
| @@ -18,6 +18,7 @@ import activityPub from './activitypub'; | ||||
| import webFinger from './webfinger'; | ||||
| import config from '../config'; | ||||
| import { updateNetworkStats } from '../services/update-chart'; | ||||
| import apiServer from './api'; | ||||
|  | ||||
| // Init app | ||||
| const app = new Koa(); | ||||
| @@ -47,7 +48,7 @@ if (config.url.startsWith('https')) { | ||||
| 	}); | ||||
| } | ||||
|  | ||||
| app.use(mount('/api', require('./api'))); | ||||
| app.use(mount('/api', apiServer)); | ||||
| app.use(mount('/files', require('./file'))); | ||||
|  | ||||
| // Init router | ||||
|   | ||||
| @@ -37,10 +37,8 @@ async function save(path: string, name: string, type: string, hash: string, size | ||||
| 	if (config.drive && config.drive.storage == 'minio') { | ||||
| 		const minio = new Minio.Client(config.drive.config); | ||||
|  | ||||
| 		const keyDir = `${config.drive.prefix}/${uuid.v4()}`; | ||||
| 		const key = `${keyDir}/${name}`; | ||||
| 		const thumbnailKeyDir = `${config.drive.prefix}/${uuid.v4()}`; | ||||
| 		const thumbnailKey = `${thumbnailKeyDir}/${name}.thumbnail.jpg`; | ||||
| 		const key = `${config.drive.prefix}/${uuid.v4()}`; | ||||
| 		const thumbnailKey = `${config.drive.prefix}/${uuid.v4()}`; | ||||
|  | ||||
| 		const baseUrl = config.drive.baseUrl | ||||
| 			|| `${ config.drive.config.useSSL ? 'https' : 'http' }://${ config.drive.config.endPoint }${ config.drive.config.port ? `:${config.drive.config.port}` : '' }/${ config.drive.bucket }`; | ||||
| @@ -64,8 +62,8 @@ async function save(path: string, name: string, type: string, hash: string, size | ||||
| 				key: key, | ||||
| 				thumbnailKey: thumbnailKey | ||||
| 			}, | ||||
| 			url: `${ baseUrl }/${ keyDir }/${ encodeURIComponent(name) }`, | ||||
| 			thumbnailUrl: thumbnail ? `${ baseUrl }/${ thumbnailKeyDir }/${ encodeURIComponent(name) }.thumbnail.jpg` : null | ||||
| 			url: `${ baseUrl }/${ key }`, | ||||
| 			thumbnailUrl: thumbnail ? `${ baseUrl }/${ thumbnailKey }` : null | ||||
| 		}); | ||||
|  | ||||
| 		const file = await DriveFile.insert({ | ||||
|   | ||||
| @@ -10,13 +10,13 @@ import renderAccept from '../../remote/activitypub/renderer/accept'; | ||||
| import { deliver } from '../../queue'; | ||||
| import createFollowRequest from './requests/create'; | ||||
|  | ||||
| export default async function(follower: IUser, followee: IUser) { | ||||
| export default async function(follower: IUser, followee: IUser, requestId?: string) { | ||||
| 	// フォロー対象が鍵アカウントである or | ||||
| 	// フォロワーがBotであり、フォロー対象がBotからのフォローに慎重である or | ||||
| 	// フォロワーがローカルユーザーであり、フォロー対象がリモートユーザーである | ||||
| 	// 上記のいずれかに当てはまる場合はすぐフォローせずにフォローリクエストを発行しておく | ||||
| 	if (followee.isLocked || (followee.carefulBot && follower.isBot) || (isLocalUser(follower) && isRemoteUser(followee))) { | ||||
| 		await createFollowRequest(follower, followee); | ||||
| 		await createFollowRequest(follower, followee, requestId); | ||||
| 		return; | ||||
| 	} | ||||
|  | ||||
| @@ -79,7 +79,7 @@ export default async function(follower: IUser, followee: IUser) { | ||||
| 	} | ||||
|  | ||||
| 	if (isRemoteUser(follower) && isLocalUser(followee)) { | ||||
| 		const content = pack(renderAccept(renderFollow(follower, followee))); | ||||
| 		const content = pack(renderAccept(renderFollow(follower, followee, requestId), followee)); | ||||
| 		deliver(followee, content, follower.inbox); | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -29,7 +29,12 @@ export default async function(followee: IUser, follower: IUser) { | ||||
| 	}); | ||||
|  | ||||
| 	if (isRemoteUser(follower)) { | ||||
| 		const content = pack(renderAccept(renderFollow(follower, followee))); | ||||
| 		const request = await FollowRequest.findOne({ | ||||
| 			followeeId: followee._id, | ||||
| 			followerId: follower._id | ||||
| 		}); | ||||
|  | ||||
| 		const content = pack(renderAccept(renderFollow(follower, followee, request.requestId), followee as ILocalUser)); | ||||
| 		deliver(followee as ILocalUser, content, follower.inbox); | ||||
| 	} | ||||
|  | ||||
|   | ||||
| @@ -6,11 +6,12 @@ import renderFollow from '../../../remote/activitypub/renderer/follow'; | ||||
| import { deliver } from '../../../queue'; | ||||
| import FollowRequest from '../../../models/follow-request'; | ||||
|  | ||||
| export default async function(follower: IUser, followee: IUser) { | ||||
| export default async function(follower: IUser, followee: IUser, requestId?: string) { | ||||
| 	await FollowRequest.insert({ | ||||
| 		createdAt: new Date(), | ||||
| 		followerId: follower._id, | ||||
| 		followeeId: followee._id, | ||||
| 		requestId, | ||||
|  | ||||
| 		// 非正規化 | ||||
| 		_follower: { | ||||
|   | ||||
| @@ -8,7 +8,12 @@ import { publishMainStream } from '../../../stream'; | ||||
|  | ||||
| export default async function(followee: IUser, follower: IUser) { | ||||
| 	if (isRemoteUser(follower)) { | ||||
| 		const content = pack(renderReject(renderFollow(follower, followee))); | ||||
| 		const request = await FollowRequest.findOne({ | ||||
| 			followeeId: followee._id, | ||||
| 			followerId: follower._id | ||||
| 		}); | ||||
|  | ||||
| 		const content = pack(renderReject(renderFollow(follower, followee, request.requestId), followee as ILocalUser)); | ||||
| 		deliver(followee as ILocalUser, content, follower.inbox); | ||||
| 	} | ||||
|  | ||||
|   | ||||
| @@ -26,7 +26,7 @@ export default async (user: IUser, note: INote, reaction: string) => new Promise | ||||
| 	} catch (e) { | ||||
| 		// duplicate key error | ||||
| 		if (e.code === 11000) { | ||||
| 			return res(null); | ||||
| 			return rej('already reacted'); | ||||
| 		} | ||||
|  | ||||
| 		console.error(e); | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| import * as mongo from 'mongodb'; | ||||
| import isObjectId from '../../misc/is-objectid'; | ||||
| import { publishMainStream } from '../../stream'; | ||||
| import User from '../../models/user'; | ||||
| import NoteUnread from '../../models/note-unread'; | ||||
| @@ -11,11 +12,11 @@ export default ( | ||||
| 	note: string | mongo.ObjectID | ||||
| ) => new Promise<any>(async (resolve, reject) => { | ||||
|  | ||||
| 	const userId: mongo.ObjectID = mongo.ObjectID.prototype.isPrototypeOf(user) | ||||
| 	const userId: mongo.ObjectID = isObjectId(user) | ||||
| 		? user as mongo.ObjectID | ||||
| 		: new mongo.ObjectID(user); | ||||
|  | ||||
| 	const noteId: mongo.ObjectID = mongo.ObjectID.prototype.isPrototypeOf(note) | ||||
| 	const noteId: mongo.ObjectID = isObjectId(note) | ||||
| 		? note as mongo.ObjectID | ||||
| 		: new mongo.ObjectID(note); | ||||
|  | ||||
|   | ||||
							
								
								
									
										30
									
								
								src/tools/clean-remote-files.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								src/tools/clean-remote-files.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,30 @@ | ||||
| import * as promiseLimit from 'promise-limit'; | ||||
| import DriveFile, { IDriveFile } from '../models/drive-file'; | ||||
| import del from '../services/drive/delete-file'; | ||||
|  | ||||
| const limit = promiseLimit(16); | ||||
|  | ||||
| DriveFile.find({ | ||||
| 	'metadata._user.host': { | ||||
| 		$ne: null | ||||
| 	}, | ||||
| 	'metadata.deletedAt': { $exists: false } | ||||
| }, { | ||||
| 	fields: { | ||||
| 		_id: true | ||||
| 	} | ||||
| }).then(async files => { | ||||
| 	console.log(`there is ${files.length} files`); | ||||
|  | ||||
| 	await Promise.all(files.map(file => limit(() => job(file)))); | ||||
|  | ||||
| 	console.log('ALL DONE'); | ||||
| }); | ||||
|  | ||||
| async function job(file: IDriveFile): Promise<any> { | ||||
| 	file = await DriveFile.findOne({ _id: file._id }); | ||||
|  | ||||
| 	await del(file, true); | ||||
|  | ||||
| 	console.log('done', file._id); | ||||
| } | ||||
							
								
								
									
										1128
									
								
								test/api.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1128
									
								
								test/api.ts
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,3 +1,7 @@ | ||||
| /* | ||||
|  * Tests of MFM | ||||
|  */ | ||||
|  | ||||
| import * as assert from 'assert'; | ||||
|  | ||||
| import analyze from '../src/mfm/parse'; | ||||
|   | ||||
		Reference in New Issue
	
	Block a user