diff --git a/.travis.yml b/.travis.yml index 45a71d83c457c3f9c99da9c541fbd54487da59dc..a91d70cf5889c9867471e365f9ebbafb09e857f1 100644 --- a/.travis.yml +++ b/.travis.yml @@ -14,6 +14,7 @@ addons: postgresql: 9.4 rvm: + - 2.3.4 - 2.4.1 services: diff --git a/Gemfile b/Gemfile index 57600ed9f84ac889cf1eec23f40f9bc0130b9c1d..0165219d3779a2f470c627d665f785f023897a48 100644 --- a/Gemfile +++ b/Gemfile @@ -1,7 +1,7 @@ # frozen_string_literal: true source 'https://rubygems.org' -ruby '2.4.1' +ruby '>= 2.3.0', '< 2.5.0' gem 'pkg-config' @@ -88,7 +88,7 @@ group :development do gem 'bullet' gem 'active_record_query_trace' - gem 'capistrano' + gem 'capistrano', '3.8.0' gem 'capistrano-rails' gem 'capistrano-rbenv' gem 'capistrano-yarn' diff --git a/Gemfile.lock b/Gemfile.lock index 2f55abec713d62f6572ec4d1cdb9aa7bcd6d0388..4b4dd105cc3179b6404b1135c5117474e2f53f67 100644 --- a/Gemfile.lock +++ b/Gemfile.lock @@ -41,7 +41,7 @@ GEM tzinfo (~> 1.1) addressable (2.5.1) public_suffix (~> 2.0, >= 2.0.2) - airbrussh (1.1.2) + airbrussh (1.2.0) sshkit (>= 1.6.1, != 1.7.0) arel (7.1.4) ast (2.3.0) @@ -469,7 +469,7 @@ DEPENDENCIES binding_of_caller browserify-rails bullet - capistrano + capistrano (= 3.8.0) capistrano-faster-assets (~> 1.0) capistrano-rails capistrano-rbenv diff --git a/ISSUE_TEMPLATE.md b/ISSUE_TEMPLATE.md index 8394b242402f3995d59b45795f946eacd46b1f97..c78bcb492a24d8ba8d5a1eeddf0eaea048b7ae56 100644 --- a/ISSUE_TEMPLATE.md +++ b/ISSUE_TEMPLATE.md @@ -3,3 +3,4 @@ * * * * - [ ] I searched or browsed the repo’s other issues to ensure this is not a duplicate. +- [ ] This bug happens on a [tagged release](https://github.com/tootsuite/mastodon/releases) and not on `master` (If you're a user, don't worry about this). diff --git a/README.md b/README.md index 0a28170b7d76c2f55a904b91e93e0573d1186574..ea7b20a5c8f2c97a6a11311e0b37f8b6c75f2a5a 100644 --- a/README.md +++ b/README.md @@ -48,6 +48,14 @@ If you would like, you can [support the development of this project on Patreon][ - **Deployable via Docker** You don't need to mess with dependencies and configuration if you want to try Mastodon, if you have Docker and Docker Compose the deployment is extremely easy +## Checking out + +If you want a stable release for production use, you should use tagged releases. To checkout the latest available tagged version: + + git clone https://github.com/tootsuite/mastodon.git + cd mastodon + git checkout $(git describe --tags `git rev-list --tags --max-count=1`) + ## Configuration - `LOCAL_DOMAIN` should be the domain/hostname of your instance. This is **absolutely required** as it is used for generating unique IDs for everything federation-related diff --git a/app/assets/javascripts/components/features/ui/containers/status_list_container.jsx b/app/assets/javascripts/components/features/ui/containers/status_list_container.jsx index f249240d827a61b47b0b2cf5c84cd82db9b0bf7b..4c33f2b61b8f6ffc429951f872af50159e01f2c5 100644 --- a/app/assets/javascripts/components/features/ui/containers/status_list_container.jsx +++ b/app/assets/javascripts/components/features/ui/containers/status_list_container.jsx @@ -24,8 +24,10 @@ const makeGetStatusIds = () => createSelector([ if (columnSettings.getIn(['regex', 'body'], '').trim().length > 0) { try { - const regex = new RegExp(columnSettings.getIn(['regex', 'body']).trim(), 'i'); - showStatus = showStatus && !regex.test(statusForId.get('reblog') ? statuses.getIn([statusForId.get('reblog'), 'content']) : statusForId.get('content')); + if (showStatus) { + const regex = new RegExp(columnSettings.getIn(['regex', 'body']).trim(), 'i'); + showStatus = !regex.test(statusForId.get('reblog') ? statuses.getIn([statusForId.get('reblog'), 'unescaped_content']) : statusForId.get('unescaped_content')); + } } catch(e) { // Bad regex, don't affect filters } diff --git a/app/assets/javascripts/components/locales/fr.jsx b/app/assets/javascripts/components/locales/fr.jsx index 8838c264f036b3b8eaa36b934835b683c11ce52b..0a1dd38aef166d5731578e45c101c970cd1c124f 100644 --- a/app/assets/javascripts/components/locales/fr.jsx +++ b/app/assets/javascripts/components/locales/fr.jsx @@ -75,6 +75,7 @@ const fr = { "navigation_bar.favourites": "Favoris", "navigation_bar.info": "Plus d'informations", "navigation_bar.logout": "Déconnexion", + "navigation_bar.mutes": "Utilisateurs muets", "navigation_bar.follow_requests": "Demandes de suivi", "reply_indicator.cancel": "Annuler", "search.placeholder": "Rechercher", diff --git a/app/assets/javascripts/components/locales/ja.jsx b/app/assets/javascripts/components/locales/ja.jsx index c64d7ecc617a1d83b109a4745bf5e0634fa07c28..79defd3256c6f40ec3499e6f6f46ca99f7430101 100644 --- a/app/assets/javascripts/components/locales/ja.jsx +++ b/app/assets/javascripts/components/locales/ja.jsx @@ -1,121 +1,125 @@ const ja = { - "column_back_button.label": "戻る", - "lightbox.close": "é–‰ã˜ã‚‹", - "loading_indicator.label": "èªã¿è¾¼ã¿ä¸...", - "status.mention": "@{name} ã•ã‚“ã¸ã®è¿”ä¿¡", - "status.delete": "削除", - "status.reply": "返信", - "status.reblog": "ブースト", - "status.favourite": "ãŠæ°—ã«å…¥ã‚Š", - "status.reblogged_by": "{name} ã•ã‚“ã«ãƒ–ーストã•ã‚Œã¾ã—ãŸ", - "status.sensitive_warning": "ä¸é©åˆ‡ãªã‚³ãƒ³ãƒ†ãƒ³ãƒ„", - "status.sensitive_toggle": "クリックã—ã¦è¡¨ç¤º", - "status.show_more": "ã‚‚ã£ã¨è¦‹ã‚‹", - "status.load_more": "ã‚‚ã£ã¨è¦‹ã‚‹", - "status.show_less": "éš ã™", - "status.open": "Expand this status", - "status.report": "@{name} ã•ã‚“ã‚’é€šå ±", - "status.media_hidden": "éžè¡¨ç¤ºã®ãƒ¡ãƒ‡ã‚¤ã‚¢", - "video_player.toggle_sound": "音ã®åˆ‡ã‚Šæ›¿ãˆ", - "account.mention": "@{name} ã•ã‚“ã«è¿”ä¿¡", - "account.edit_profile": "プãƒãƒ•ã‚£ãƒ¼ãƒ«ã‚’編集", - "account.unblock": "@{name} ã•ã‚“ã®ãƒ–ãƒãƒƒã‚¯ã‚’解除", - "account.unfollow": "フォãƒãƒ¼è§£é™¤", "account.block": "@{name} ã•ã‚“をブãƒãƒƒã‚¯", - "account.mute": "ミュート", - "account.unmute": "ミュート解除", + "account.disclaimer": "ã“ã®ãƒ¦ãƒ¼ã‚¶ãƒ¼ã¯ä»–ã®ã‚¤ãƒ³ã‚¹ã‚¿ãƒ³ã‚¹ã«æ‰€å±žã—ã¦ã„ã‚‹ãŸã‚ã€æ•°å—ãŒæ£ç¢ºã§ç„¡ã„å ´åˆãŒã‚ã‚Šã¾ã™ã€‚", + "account.edit_profile": "プãƒãƒ•ã‚£ãƒ¼ãƒ«ã‚’編集", "account.follow": "フォãƒãƒ¼", - "account.report": "@{name}ã‚’é€šå ±ã™ã‚‹", - "account.posts": "投稿", - "account.follows": "フォãƒãƒ¼", "account.followers": "フォãƒãƒ¯ãƒ¼", + "account.follows": "フォãƒãƒ¼", "account.follows_you": "フォãƒãƒ¼ã•ã‚Œã¦ã„ã¾ã™", + "account.mention": "@{name} ã•ã‚“ã«è¿”ä¿¡", + "account.mute": "ミュート", + "account.posts": "投稿", + "account.report": "@{name}ã‚’é€šå ±ã™ã‚‹", "account.requested": "承èªå¾…ã¡", - "follow_request.authorize": "許å¯", - "follow_request.reject": "æ‹’å¦", - "getting_started.heading": "スタート", - "getting_started.about_addressing": "ドメインã¨ãƒ¦ãƒ¼ã‚¶ãƒ¼åを知ã£ã¦ã„ã‚‹ãªã‚‰æ¤œç´¢ãƒ•ã‚©ãƒ¼ãƒ ã«å…¥åŠ›ã™ã‚Œã°ãƒ•ã‚©ãƒãƒ¼ã§ãã¾ã™ã€‚", - "getting_started.about_shortcuts": "対象ã®ã‚¢ã‚«ã‚¦ãƒ³ãƒˆãŒã‚ãªãŸã¨åŒã˜ãƒ‰ãƒ¡ã‚¤ãƒ³ã®ãƒ¦ãƒ¼ã‚¶ãƒ¼ãªã‚‰ã°ãƒ¦ãƒ¼ã‚¶ãƒ¼åã®ã¿ã§æ¤œç´¢ã§ãã¾ã™ã€‚ã“ã‚Œã¯è¿”ä¿¡ã®ã¨ãも一緒ã§ã™ã€‚", - "getting_started.open_source_notice": "Mastodon ã¯ã‚ªãƒ¼ãƒ—ンソースソフトウェアã§ã™ã€‚誰ã§ã‚‚ GitHub({github})ã‹ã‚‰é–‹ç™ºã«å‚åŠ ã—ãŸã‚Šã€å•é¡Œã‚’å ±å‘Šã—ãŸã‚Šã§ãã¾ã™ã€‚ {apps}", - "getting_started.apps": "ã•ã¾ã–ã¾ãªã‚¢ãƒ—リã§åˆ©ç”¨ã§ãã¾ã™ã€‚", - "column.home": "ホーム", + "account.unblock": "@{name} ã•ã‚“ã®ãƒ–ãƒãƒƒã‚¯ã‚’解除", + "account.unfollow": "フォãƒãƒ¼è§£é™¤", + "account.unmute": "ミュート解除", + "boost_modal.combo": "次ã‹ã‚‰ã¯{combo}を押ã›ã°ã€ã“れをスã‚ップã§ãã¾ã™ã€‚", + "column.blocks": "ブãƒãƒƒã‚¯ã—ãŸãƒ¦ãƒ¼ã‚¶ãƒ¼", "column.community": "ãƒãƒ¼ã‚«ãƒ«ã‚¿ã‚¤ãƒ ライン", - "column.public": "連åˆã‚¿ã‚¤ãƒ ライン", - "column.notifications": "通知", "column.favourites": "ãŠæ°—ã«å…¥ã‚Š", - "tabs_bar.compose": "投稿", - "tabs_bar.home": "ホーム", - "tabs_bar.mentions": "返信", - "tabs_bar.local_timeline": "ãƒãƒ¼ã‚«ãƒ«", - "tabs_bar.federated_timeline": "連åˆ", - "tabs_bar.notifications": "通知", + "column.follow_requests": "フォãƒãƒ¼ãƒªã‚¯ã‚¨ã‚¹ãƒˆ", + "column.home": "ホーム", + "column.mutes": "ミュートã—ãŸãƒ¦ãƒ¼ã‚¶ãƒ¼", + "column.notifications": "通知", + "column.public": "連åˆã‚¿ã‚¤ãƒ ライン", + "column_back_button.label": "戻る", "compose_form.placeholder": "今ãªã«ã—ã¦ã‚‹ï¼Ÿ", + "compose_form.privacy_disclaimer": "ã‚ãªãŸã®éžå…¬é–‹ãƒˆã‚¥ãƒ¼ãƒˆã¯è¿”ä¿¡å…ˆã®ãƒ¦ãƒ¼ã‚¶ãƒ¼ï¼ˆat {domains})ã«å…¬é–‹ã•ã‚Œã¾ã™ã€‚{domainsCount, plural, one {that server} other {those servers}}ã‚’ä¿¡é ¼ã—ã¾ã™ã‹ï¼ŸæŠ•ç¨¿ã®ãƒ—ライãƒã‚·ãƒ¼ä¿è·ã¯Mastodonサーãƒãƒ¼å†…ã§ã®ã¿æœ‰åŠ¹ã§ã™ã€‚ ã‚‚ã—{domains} {domainsCount, plural, one {is not a Mastodon instance} other {are not Mastodon instances}}ãªã‚‰ã°ã‚ãªãŸã®æŠ•ç¨¿ã®ãƒ—ライãƒã‚·ãƒ¼ã¯ä¿è·ã•ã‚Œãšã€ãƒ–ーストã•ã‚ŒãŸã‚ŠäºˆæœŸã—ãªã„ユーザーã«è¦‹ã‚‰ã‚Œã‚‹å¯èƒ½æ€§ãŒã‚ã‚Šã¾ã™ã€‚", "compose_form.publish": "トゥート", "compose_form.sensitive": "メディアをä¸é©åˆ‡ãªã‚³ãƒ³ãƒ†ãƒ³ãƒ„ã¨ã—ã¦ãƒžãƒ¼ã‚¯ã™ã‚‹", "compose_form.spoiler": "テã‚ã‚¹ãƒˆã‚’éš ã™", - "compose_form.spoiler_placeholder": "内容注æ„メッセージ", - "compose_form.private": "éžå…¬é–‹ã«ã™ã‚‹", - "compose_form.privacy_disclaimer": "ã‚ãªãŸã®éžå…¬é–‹ãƒˆã‚¥ãƒ¼ãƒˆã¯è¿”ä¿¡å…ˆã®ãƒ¦ãƒ¼ã‚¶ãƒ¼ï¼ˆat {domains})ã«å…¬é–‹ã•ã‚Œã¾ã™ã€‚{domainsCount, plural, one {that server} other {those servers}}ã‚’ä¿¡é ¼ã—ã¾ã™ã‹ï¼ŸæŠ•ç¨¿ã®ãƒ—ライãƒã‚·ãƒ¼ä¿è·ã¯Mastodonサーãƒãƒ¼å†…ã§ã®ã¿æœ‰åŠ¹ã§ã™ã€‚ ã‚‚ã—{domains} {domainsCount, plural, one {is not a Mastodon instance} other {are not Mastodon instances}}ãªã‚‰ã°ã‚ãªãŸã®æŠ•ç¨¿ã®ãƒ—ライãƒã‚·ãƒ¼ã¯ä¿è·ã•ã‚Œãšã€ãƒ–ーストã•ã‚ŒãŸã‚ŠäºˆæœŸã—ãªã„ユーザーã«è¦‹ã‚‰ã‚Œã‚‹å¯èƒ½æ€§ãŒã‚ã‚Šã¾ã™ã€‚", - "compose_form.unlisted": "公開タイムラインã«è¡¨ç¤ºã—ãªã„", - "privacy.public.short": "公開", - "privacy.public.long": "公開TLã«æŠ•ç¨¿ã™ã‚‹", - "privacy.unlisted.short": "未åŽè¼‰", - "privacy.unlisted.long": "公開TLã§è¡¨ç¤ºã—ãªã„", - "privacy.private.short": "éžå…¬é–‹", - "privacy.private.long": "フォãƒãƒ¯ãƒ¼ã ã‘ã«å…¬é–‹", - "privacy.direct.short": "ダイレクト", - "privacy.direct.long": "å«ã‚“ã ユーザーã ã‘ã«å…¬é–‹", - "privacy.change": "投稿ã®ãƒ—ライãƒã‚·ãƒ¼ã‚’変更", - "report.heading": "æ–°è¦é€šå ±", - "report.placeholder": "コメント", - "report.target": "å•é¡Œã®ãƒ¦ãƒ¼ã‚¶ãƒ¼", - "report.submit": "é€šå ±ã™ã‚‹", - "navigation_bar.edit_profile": "プãƒãƒ•ã‚£ãƒ¼ãƒ«ã‚’編集", - "navigation_bar.preferences": "ユーザーè¨å®š", + "compose_form.spoiler_placeholder": "閲覧注æ„", + "emoji_button.label": "絵文å—ã‚’è¿½åŠ ", + "empty_column.community": "ãƒãƒ¼ã‚«ãƒ«ã‚¿ã‚¤ãƒ ラインã¯ã¾ã 使ã‚ã‚Œã¦ã„ã¾ã›ã‚“。何ã‹æ›¸ã„ã¦ã¿ã¾ã—ょã†ï¼", + "empty_column.hashtag": "ã“ã®ãƒãƒƒã‚·ãƒ¥ã‚¿ã‚°ã¯ã¾ã 使ã‚ã‚Œã¦ã„ã¾ã›ã‚“。", + "empty_column.home": "ã¾ã 誰もフォãƒãƒ¼ã—ã¦ã„ã¾ã›ã‚“。{public}を見ã«è¡Œãã‹ã€æ¤œç´¢ã‚’使ã£ã¦ä»–ã®ãƒ¦ãƒ¼ã‚¶ãƒ¼ã‚’見ã¤ã‘ã¾ã—ょã†ã€‚", + "empty_column.home.public_timeline": "連åˆã‚¿ã‚¤ãƒ ライン", + "empty_column.notifications": "ã¾ã 通知ãŒã‚ã‚Šã¾ã›ã‚“。他ã®äººã¨ãµã‚Œåˆã£ã¦ä¼šè©±ã‚’始ã‚ã¾ã—ょã†ã€‚", + "empty_column.public": "ã“ã“ã«ã¯ã¾ã 何もã‚ã‚Šã¾ã›ã‚“ï¼å…¬é–‹ã§ä½•ã‹ã‚’投稿ã—ãŸã‚Šã€ä»–ã®ã‚¤ãƒ³ã‚¹ã‚¿ãƒ³ã‚¹ã®ãƒ¦ãƒ¼ã‚¶ãƒ¼ã‚’フォãƒãƒ¼ã—ãŸã‚Šã—ã¦ã„ã£ã±ã„ã«ã—ã¾ã—ょã†ï¼", + "follow_request.authorize": "許å¯", + "follow_request.reject": "æ‹’å¦", + "getting_started.apps": "ã•ã¾ã–ã¾ãªã‚¢ãƒ—リã§åˆ©ç”¨ã§ãã¾ã™ã€‚", + "getting_started.heading": "スタート", + "getting_started.open_source_notice": "Mastodon ã¯ã‚ªãƒ¼ãƒ—ンソースソフトウェアã§ã™ã€‚誰ã§ã‚‚ GitHub({github})ã‹ã‚‰é–‹ç™ºã«å‚åŠ ã—ãŸã‚Šã€å•é¡Œã‚’å ±å‘Šã—ãŸã‚Šã§ãã¾ã™ã€‚ {apps}", + "home.column_settings.advanced": "上級者å‘ã‘", + "home.column_settings.basic": "シンプル", + "home.column_settings.filter_regex": "æ£è¦è¡¨ç¾ã§ãƒ•ã‚£ãƒ«ã‚¿ãƒ¼", + "home.column_settings.show_reblogs": "ブースト表示", + "home.column_settings.show_replies": "返信表示", + "home.settings": "カラムè¨å®š", + "lightbox.close": "é–‰ã˜ã‚‹", + "loading_indicator.label": "èªã¿è¾¼ã¿ä¸...", + "media_gallery.toggle_visible": "表示切り替ãˆ", + "missing_indicator.label": "見ã¤ã‹ã‚Šã¾ã›ã‚“", + "navigation_bar.blocks": "ブãƒãƒƒã‚¯ã—ãŸãƒ¦ãƒ¼ã‚¶ãƒ¼", "navigation_bar.community_timeline": "ãƒãƒ¼ã‚«ãƒ«ã‚¿ã‚¤ãƒ ライン", - "navigation_bar.public_timeline": "連åˆã‚¿ã‚¤ãƒ ライン", - "navigation_bar.logout": "ãƒã‚°ã‚¢ã‚¦ãƒˆ", + "navigation_bar.edit_profile": "プãƒãƒ•ã‚£ãƒ¼ãƒ«ã‚’編集", "navigation_bar.favourites": "ãŠæ°—ã«å…¥ã‚Š", - "navigation_bar.blocks": "ブãƒãƒƒã‚¯ã—ãŸãƒ¦ãƒ¼ã‚¶ãƒ¼", + "navigation_bar.follow_requests": "フォãƒãƒ¼ãƒªã‚¯ã‚¨ã‚¹ãƒˆ", "navigation_bar.info": "サーãƒãƒ¼æƒ…å ±", - "reply_indicator.cancel": "ã‚ャンセル", - "search.placeholder": "検索", - "search.account": "アカウント", - "search.hashtag": "ãƒãƒƒã‚·ãƒ¥ã‚¿ã‚°", - "search.status_by": "{uuuname}ã‹ã‚‰ã®æŠ•ç¨¿", - "search_results.total": "{count} 件", - "upload_area.title": "ファイルをã“ã¡ã‚‰ã«ãƒ‰ãƒ©ãƒƒã‚°ã—ã¦ãã ã•ã„", - "upload_button.label": "ãƒ¡ãƒ‡ã‚£ã‚¢ã‚’è¿½åŠ ", - "upload_form.undo": "ã‚„ã‚Šç›´ã™", - "notification.follow": "{name} ã•ã‚“ã«ãƒ•ã‚©ãƒãƒ¼ã•ã‚Œã¾ã—ãŸ", + "navigation_bar.logout": "ãƒã‚°ã‚¢ã‚¦ãƒˆ", + "navigation_bar.mutes": "ミュートã—ãŸãƒ¦ãƒ¼ã‚¶ãƒ¼", + "navigation_bar.preferences": "ユーザーè¨å®š", + "navigation_bar.public_timeline": "連åˆã‚¿ã‚¤ãƒ ライン", "notification.favourite": "{name} ã•ã‚“ãŒã‚ãªãŸã®ãƒˆã‚¥ãƒ¼ãƒˆã‚’ãŠæ°—ã«å…¥ã‚Šã«ç™»éŒ²ã—ã¾ã—ãŸ", - "notification.reblog": "{name} ã•ã‚“ãŒã‚ãªãŸã®ãƒˆã‚¥ãƒ¼ãƒˆã‚’ブーストã—ã¾ã—ãŸ", + "notification.follow": "{name} ã•ã‚“ã«ãƒ•ã‚©ãƒãƒ¼ã•ã‚Œã¾ã—ãŸ", "notification.mention": "{name} ã•ã‚“ãŒã‚ãªãŸã«è¿”ä¿¡ã—ã¾ã—ãŸ", - "notifications.clear": "通知を片付ã‘ã‚‹", - "notifications.clear_confirmation": "通知を全部片付ã‘ã¾ã™ã€‚大丈夫ã§ã™ã‹ï¼Ÿ", + "notification.reblog": "{name} ã•ã‚“ãŒã‚ãªãŸã®ãƒˆã‚¥ãƒ¼ãƒˆã‚’ブーストã—ã¾ã—ãŸ", + "notifications.clear": "通知を消去", + "notifications.clear_confirmation": "本当ã«é€šçŸ¥ã‚’消去ã—ã¾ã™ã‹ï¼Ÿ", "notifications.column_settings.alert": "デスクトップ通知", - "notifications.column_settings.show": "カラムã«è¡¨ç¤º", - "notifications.column_settings.follow": "æ–°ã—ã„フォãƒãƒ¯ãƒ¼", "notifications.column_settings.favourite": "ãŠæ°—ã«å…¥ã‚Š", + "notifications.column_settings.follow": "æ–°ã—ã„フォãƒãƒ¯ãƒ¼", "notifications.column_settings.mention": "返信", "notifications.column_settings.reblog": "ブースト", + "notifications.column_settings.show": "カラムã«è¡¨ç¤º", "notifications.column_settings.sound": "通知音をå†ç”Ÿ", - "empty_column.home": "ã¾ã 誰もフォãƒãƒ¼ã—ã¦ã„ã¾ã›ã‚“。{public}を見ã«è¡Œãã‹ã€æ¤œç´¢ã‚’使ã£ã¦ä»–ã®ãƒ¦ãƒ¼ã‚¶ãƒ¼ã‚’見ã¤ã‘ã¾ã—ょã†ã€‚", - "empty_column.home.public_timeline": "連åˆã‚¿ã‚¤ãƒ ライン", - "empty_column.notifications": "ã¾ã 通知ãŒã‚ã‚Šã¾ã›ã‚“。他ã®äººã¨ãµã‚Œåˆã£ã¦ä¼šè©±ã‚’始ã‚ã¾ã—ょã†ã€‚", - "empty_column.public": "ã“ã“ã«ã¯ã¾ã 何もã‚ã‚Šã¾ã›ã‚“ï¼å…¬é–‹ã§ä½•ã‹ã‚’投稿ã—ãŸã‚Šã€ä»–ã®ã‚¤ãƒ³ã‚¹ã‚¿ãƒ³ã‚¹ã®ãƒ¦ãƒ¼ã‚¶ãƒ¼ã‚’フォãƒãƒ¼ã—ãŸã‚Šã—ã¦ã„ã£ã±ã„ã«ã—ã¾ã—ょã†ï¼", - "empty_column.hashtag": "ã“ã®ãƒãƒƒã‚·ãƒ¥ã‚¿ã‚°ã¯ã¾ã 使ã£ã¦ã„ã¾ã›ã‚“。", - "upload_progress.label": "アップãƒãƒ¼ãƒ‰ä¸â€¦", - "emoji_button.label": "絵文å—ã‚’è¿½åŠ ", - "home.column_settings.basic": "シンプル", - "home.column_settings.advanced": "エã‚スパート", - "home.column_settings.show_reblogs": "ブースト表示", - "home.column_settings.show_replies": "返信表示", - "home.column_settings.filter_regex": "æ£è¦è¡¨ç¾ã§ãƒ•ã‚£ãƒ«ã‚¿ãƒ¼", - "home.settings": "カラムè¨å®š", "notifications.settings": "カラムè¨å®š", - "missing_indicator.label": "見ã¤ã‹ã‚Šã¾ã›ã‚“", - "boost_modal.combo": "次ã¯{combo}を押ã›ã°ã€ã“れをスã‚ップã§ãã¾ã™ã€‚" + "privacy.change": "投稿ã®ãƒ—ライãƒã‚·ãƒ¼ã‚’変更", + "privacy.direct.long": "メンションã—ãŸãƒ¦ãƒ¼ã‚¶ãƒ¼ã ã‘ã«å…¬é–‹", + "privacy.direct.short": "ダイレクト", + "privacy.private.long": "フォãƒãƒ¯ãƒ¼ã ã‘ã«å…¬é–‹", + "privacy.private.short": "éžå…¬é–‹", + "privacy.public.long": "公開TLã«æŠ•ç¨¿ã™ã‚‹", + "privacy.public.short": "公開", + "privacy.unlisted.long": "公開TLã§è¡¨ç¤ºã—ãªã„", + "privacy.unlisted.short": "未åŽè¼‰", + "reply_indicator.cancel": "ã‚ャンセル", + "report.heading": "æ–°è¦é€šå ±", + "report.placeholder": "コメント", + "report.submit": "é€šå ±ã™ã‚‹", + "report.target": "å•é¡Œã®ãƒ¦ãƒ¼ã‚¶ãƒ¼", + "search.placeholder": "検索", + "search.status_by": "{name}ã‹ã‚‰ã®æŠ•ç¨¿", + "search_results.total": "{count} {count, plural, one {result} other {results}} 件", + "status.delete": "削除", + "status.favourite": "ãŠæ°—ã«å…¥ã‚Š", + "status.load_more": "ã‚‚ã£ã¨è¦‹ã‚‹", + "status.media_hidden": "éžè¡¨ç¤ºã®ãƒ¡ãƒ‡ã‚¤ã‚¢", + "status.mention": "@{name} ã•ã‚“ã¸ã®è¿”ä¿¡", + "status.open": "詳細を表示", + "status.reblog": "ブースト", + "status.reblogged_by": "{name} ã•ã‚“ã«ãƒ–ーストã•ã‚Œã¾ã—ãŸ", + "status.reply": "返信", + "status.report": "@{name} ã•ã‚“ã‚’é€šå ±", + "status.sensitive_toggle": "クリックã—ã¦è¡¨ç¤º", + "status.sensitive_warning": "ä¸é©åˆ‡ãªã‚³ãƒ³ãƒ†ãƒ³ãƒ„", + "status.show_less": "éš ã™", + "status.show_more": "ã‚‚ã£ã¨è¦‹ã‚‹", + "tabs_bar.compose": "投稿", + "tabs_bar.federated_timeline": "連åˆ", + "tabs_bar.home": "ホーム", + "tabs_bar.local_timeline": "ãƒãƒ¼ã‚«ãƒ«", + "tabs_bar.notifications": "通知", + "upload_area.title": "ドラッグ&ドãƒãƒƒãƒ—ã§ã‚¢ãƒƒãƒ—ãƒãƒ¼ãƒ‰", + "upload_button.label": "ãƒ¡ãƒ‡ã‚£ã‚¢ã‚’è¿½åŠ ", + "upload_form.undo": "ã‚„ã‚Šç›´ã™", + "upload_progress.label": "アップãƒãƒ¼ãƒ‰ä¸â€¦", + "video_player.expand": "å‹•ç”»ã®è©³ç´°", + "video_player.toggle_sound": "音ã®åˆ‡ã‚Šæ›¿ãˆ", + "video_player.toggle_visible": "表示切り替ãˆ", + "video_player.video_error": "å‹•ç”»ã®å†ç”Ÿã«å¤±æ•—ã—ã¾ã—ãŸ", }; export default ja; diff --git a/app/assets/javascripts/components/reducers/statuses.jsx b/app/assets/javascripts/components/reducers/statuses.jsx index ca8fa7a015b9049b5d45fe41d653439a43f99400..2002d22233903adea7998f4925513eecf4a7a8b7 100644 --- a/app/assets/javascripts/components/reducers/statuses.jsx +++ b/app/assets/javascripts/components/reducers/statuses.jsx @@ -48,6 +48,9 @@ const normalizeStatus = (state, status) => { normalStatus.reblog = status.reblog.id; } + const linebreakComplemented = status.content.replace(/<br \/>/g, '\n').replace(/<\/p><p>/g, '\n\n'); + normalStatus.unescaped_content = new DOMParser().parseFromString(linebreakComplemented, 'text/html').documentElement.textContent; + return state.update(status.id, Immutable.Map(), map => map.mergeDeep(Immutable.fromJS(normalStatus))); }; diff --git a/app/assets/stylesheets/components.scss b/app/assets/stylesheets/components.scss index d1c4d2bb2bcebc05ba57a56a8a18e27ec93d9ef8..8bd35819a934541eb503725b35246ff0981d6e83 100644 --- a/app/assets/stylesheets/components.scss +++ b/app/assets/stylesheets/components.scss @@ -1203,6 +1203,10 @@ a.status__content__spoiler-link { &:focus { outline: 0; } + + @media screen and (max-width: 600px) { + font-size: 16px; + } } .spoiler-input__input { @@ -1267,6 +1271,10 @@ a.status__content__spoiler-link { color: $color5; border-bottom-color: $color4; } + + @media screen and (max-width: 600px) { + font-size: 16px; + } } @import 'boost'; @@ -1906,6 +1914,10 @@ button.icon-button.active i.fa-retweet { &:focus { background: lighten($color1, 4%); } + + @media screen and (max-width: 600px) { + font-size: 16px; + } } .search__icon { diff --git a/app/controllers/admin/domain_blocks_controller.rb b/app/controllers/admin/domain_blocks_controller.rb index a8b56c0859b3498dbee132101f80b1ecdd34138b..5d146d9469babb8c2d75983d3e87be9214058c6d 100644 --- a/app/controllers/admin/domain_blocks_controller.rb +++ b/app/controllers/admin/domain_blocks_controller.rb @@ -15,16 +15,26 @@ module Admin if @domain_block.save DomainBlockWorker.perform_async(@domain_block.id) - redirect_to admin_domain_blocks_path, notice: 'Domain block is now being processed' + redirect_to admin_domain_blocks_path, notice: I18n.t('admin.domain_block.created_msg') else render action: :new end end + def show + @domain_block = DomainBlock.find(params[:id]) + end + + def destroy + @domain_block = DomainBlock.find(params[:id]) + UnblockDomainService.new.call(@domain_block, resource_params[:retroactive]) + redirect_to admin_domain_blocks_path, notice: I18n.t('admin.domain_block.destroyed_msg') + end + private def resource_params - params.require(:domain_block).permit(:domain, :severity) + params.require(:domain_block).permit(:domain, :severity, :reject_media, :retroactive) end end end diff --git a/app/controllers/application_controller.rb b/app/controllers/application_controller.rb index 0c320177ddc4354857a55cb6bfb4577d60e24c0a..e8d7de21883288581f1f00ab7bddbb1550f1e478 100644 --- a/app/controllers/application_controller.rb +++ b/app/controllers/application_controller.rb @@ -8,7 +8,9 @@ class ApplicationController < ActionController::Base force_ssl if: "Rails.env.production? && ENV['LOCAL_HTTPS'] == 'true'" include Localized - helper_method :current_account, :single_user_mode? + + helper_method :current_account + helper_method :single_user_mode? rescue_from ActionController::RoutingError, with: :not_found rescue_from ActiveRecord::RecordNotFound, with: :not_found diff --git a/app/models/concerns/.keep b/app/models/concerns/.keep deleted file mode 100644 index e69de29bb2d1d6434b8b29ae775ad8c2e48c5391..0000000000000000000000000000000000000000 diff --git a/app/models/domain_block.rb b/app/models/domain_block.rb index 3548ccd692063e8c29e63f1f36801fbf7285acfd..89c81f766a28a6b584aec6e00dc7a89be820295c 100644 --- a/app/models/domain_block.rb +++ b/app/models/domain_block.rb @@ -3,6 +3,8 @@ class DomainBlock < ApplicationRecord enum severity: [:silence, :suspend] + attr_accessor :retroactive + validates :domain, presence: true, uniqueness: true def self.blocked?(domain) diff --git a/app/models/import.rb b/app/models/import.rb index 3013bc50eb47c9a6b06efad2bb40dc3cabcf943b..85f6ca4bdf56b36da32f49c5e8d64efa009b915b 100644 --- a/app/models/import.rb +++ b/app/models/import.rb @@ -1,13 +1,15 @@ # frozen_string_literal: true class Import < ApplicationRecord + FILE_TYPES = ['text/plain', 'text/csv'].freeze + self.inheritance_column = false - enum type: [:following, :blocking, :muting] + belongs_to :account, required: true - belongs_to :account + enum type: [:following, :blocking, :muting] - FILE_TYPES = ['text/plain', 'text/csv'].freeze + validates :type, presence: true has_attached_file :data, url: '/system/:hash.:extension', hash_secret: ENV['PAPERCLIP_SECRET'] validates_attachment_content_type :data, content_type: FILE_TYPES diff --git a/app/models/status.rb b/app/models/status.rb index 22d93947a937e0a5c88709c45d4bc01c3bd3f6e3..5393acfcc2a502c4d35b068e3a0416551b7a703f 100644 --- a/app/models/status.rb +++ b/app/models/status.rb @@ -110,6 +110,10 @@ class Status < ApplicationRecord results end + def non_sensitive_with_media? + !sensitive? && media_attachments.any? + end + class << self def as_home_timeline(account) where(account: [account] + account.following) diff --git a/app/services/block_domain_service.rb b/app/services/block_domain_service.rb index 6c131bd3410c35b05990ff2297476e89424e0a9c..97d2ebcd7674e3e08474679bda0659d9c6630961 100644 --- a/app/services/block_domain_service.rb +++ b/app/services/block_domain_service.rb @@ -3,12 +3,34 @@ class BlockDomainService < BaseService def call(domain_block) if domain_block.silence? - Account.where(domain: domain_block.domain).update_all(silenced: true) + silence_accounts!(domain_block.domain) + clear_media!(domain_block.domain) if domain_block.reject_media? else - Account.where(domain: domain_block.domain).find_each do |account| - account.subscription(api_subscription_url(account.id)).unsubscribe if account.subscribed? - SuspendAccountService.new.call(account) - end + suspend_accounts!(domain_block.domain) + end + end + + private + + def silence_accounts!(domain) + Account.where(domain: domain).update_all(silenced: true) + end + + def clear_media!(domain) + Account.where(domain: domain).find_each do |account| + account.avatar.destroy + account.header.destroy + end + + MediaAttachment.where(account: Account.where(domain: domain)).find_each do |attachment| + attachment.file.destroy + end + end + + def suspend_accounts!(domain) + Account.where(domain: domain).where(suspended: false).find_each do |account| + account.subscription(api_subscription_url(account.id)).unsubscribe if account.subscribed? + SuspendAccountService.new.call(account) end end end diff --git a/app/services/follow_remote_account_service.rb b/app/services/follow_remote_account_service.rb index dce712b403ce5637630a89d51111f194c9ce819a..14bc064d5df02a43a7ca3ad9f4d07486aafe4106 100644 --- a/app/services/follow_remote_account_service.rb +++ b/app/services/follow_remote_account_service.rb @@ -16,7 +16,7 @@ class FollowRemoteAccountService < BaseService return Account.find_local(username) if TagManager.instance.local_domain?(domain) account = Account.find_remote(username, domain) - return account unless account&.last_webfingered_at.nil? || 1.day.from_now(account.last_webfingered_at) < Time.now.utc + return account unless account_needs_webfinger_update?(account) Rails.logger.debug "Looking up webfinger for #{uri}" @@ -62,6 +62,10 @@ class FollowRemoteAccountService < BaseService private + def account_needs_webfinger_update?(account) + account&.last_webfingered_at.nil? || account.last_webfingered_at <= 1.day.ago + end + def get_feed(url) response = http_client.get(Addressable::URI.parse(url)) [response.to_s, Nokogiri::XML(response)] diff --git a/app/services/process_feed_service.rb b/app/services/process_feed_service.rb index 321f53f22e60461dc0e757757e66d9c9204489d7..fa0633b2759e0db15807d355fd6128b6e6ceafb9 100644 --- a/app/services/process_feed_service.rb +++ b/app/services/process_feed_service.rb @@ -179,12 +179,12 @@ class ProcessFeedService < BaseService end def hashtags_from_xml(parent, xml) - tags = xml.xpath('./xmlns:category', xmlns: TagManager::XMLNS).map { |category| category['term'] }.select { |t| !t.blank? } + tags = xml.xpath('./xmlns:category', xmlns: TagManager::XMLNS).map { |category| category['term'] }.select(&:present?) ProcessHashtagsService.new.call(parent, tags) end def media_from_xml(parent, xml) - return if DomainBlock.find_by(domain: parent.account.domain)&.reject_media? + do_not_download = DomainBlock.find_by(domain: parent.account.domain)&.reject_media? xml.xpath('./xmlns:link[@rel="enclosure"]', xmlns: TagManager::XMLNS).each do |link| next unless link['href'] @@ -192,7 +192,11 @@ class ProcessFeedService < BaseService media = MediaAttachment.where(status: parent, remote_url: link['href']).first_or_initialize(account: parent.account, status: parent, remote_url: link['href']) parsed_url = URI.parse(link['href']) - next if !%w(http https).include?(parsed_url.scheme) || parsed_url.host.empty? + next if !%w[http https].include?(parsed_url.scheme) || parsed_url.host.empty? + + media.save + + next if do_not_download begin media.file_remote_url = link['href'] diff --git a/app/services/suspend_account_service.rb b/app/services/suspend_account_service.rb index 42ff4dcb78c2039488003a47b477d6c24a7b37c2..66517470e6c905201650316afd03cf9a4a4ec6b9 100644 --- a/app/services/suspend_account_service.rb +++ b/app/services/suspend_account_service.rb @@ -13,6 +13,7 @@ class SuspendAccountService < BaseService def purge_content @account.statuses.reorder(nil).find_each do |status| + # This federates out deletes to previous followers RemoveStatusService.new.call(status) end @@ -29,9 +30,7 @@ class SuspendAccountService < BaseService @account.display_name = '' @account.note = '' @account.avatar.destroy - @account.avatar.clear @account.header.destroy - @account.header.clear @account.save! end diff --git a/app/services/unblock_domain_service.rb b/app/services/unblock_domain_service.rb new file mode 100644 index 0000000000000000000000000000000000000000..9794e439d14421d02cf1d308b5867efccccc0433 --- /dev/null +++ b/app/services/unblock_domain_service.rb @@ -0,0 +1,15 @@ +# frozen_string_literal: true + +class UnblockDomainService < BaseService + def call(domain_block, retroactive) + if retroactive + if domain_block.silence? + Account.where(domain: domain_block.domain).update_all(silenced: false) + else + Account.where(domain: domain_block.domain).update_all(suspended: false) + end + end + + domain_block.destroy + end +end diff --git a/app/services/unfollow_service.rb b/app/services/unfollow_service.rb index 244c9b52904f24dcda8a967ec534f45af750c898..9b39f4945a1d27adacbe9614093aeb9fa0a37f48 100644 --- a/app/services/unfollow_service.rb +++ b/app/services/unfollow_service.rb @@ -6,6 +6,7 @@ class UnfollowService < BaseService # @param [Account] target_account Which to unfollow def call(source_account, target_account) follow = source_account.unfollow!(target_account) + return unless follow NotificationWorker.perform_async(build_xml(follow), source_account.id, target_account.id) unless target_account.local? UnmergeWorker.perform_async(target_account.id, source_account.id) end diff --git a/app/views/about/terms.no.html.haml b/app/views/about/terms.no.html.haml index 5506cd863d5313a3459865512abeb122365ef81a..32ec57ed19d17a4947cd0b81c068f65104857174 100644 --- a/app/views/about/terms.no.html.haml +++ b/app/views/about/terms.no.html.haml @@ -71,6 +71,6 @@ %p Dette dokumentet er lisensiert under CC-BY-SA. De ble sist oppdatert 12. april 2017. %p - Dokumentet er en adoptert og endret versjon fra + Dokumentet er en adoptert og endret versjon fra = succeed '.' do = link_to 'Discourse privacy policy', 'https://github.com/discourse/discourse' diff --git a/app/views/accounts/_header.html.haml b/app/views/accounts/_header.html.haml index beee96cd8d773ae66118cd42160d50c869d2576d..6538885a01eb8d3c97b2cc44942f0b98a9546d88 100644 --- a/app/views/accounts/_header.html.haml +++ b/app/views/accounts/_header.html.haml @@ -1,34 +1,34 @@ -.card.h-card.p-author{ style: "background-image: url(#{@account.header.url( :original)})" } - - if user_signed_in? && current_account.id != @account.id && !current_account.requested?(@account) +.card.h-card.p-author{ style: "background-image: url(#{account.header.url( :original)})" } + - if user_signed_in? && current_account.id != account.id && !current_account.requested?(account) .controls - - if current_account.following?(@account) - = link_to t('accounts.unfollow'), unfollow_account_path(@account), data: { method: :post }, class: 'button' + - if current_account.following?(account) + = link_to t('accounts.unfollow'), unfollow_account_path(account), data: { method: :post }, class: 'button' - else - = link_to t('accounts.follow'), follow_account_path(@account), data: { method: :post }, class: 'button' + = link_to t('accounts.follow'), follow_account_path(account), data: { method: :post }, class: 'button' - elsif !user_signed_in? .controls .remote-follow - = link_to t('accounts.remote_follow'), account_remote_follow_path(@account), class: 'button' - .avatar= image_tag @account.avatar.url(:original), class: 'u-photo' + = link_to t('accounts.remote_follow'), account_remote_follow_path(account), class: 'button' + .avatar= image_tag account.avatar.url(:original), class: 'u-photo' %h1.name - %span.p-name.emojify= display_name(@account) + %span.p-name.emojify= display_name(account) %small - %span= "@#{@account.username}" - = fa_icon('lock') if @account.locked? + %span= "@#{account.username}" + = fa_icon('lock') if account.locked? .details .bio - .account__header__content.p-note.emojify= Formatter.instance.simplified_format(@account) + .account__header__content.p-note.emojify= Formatter.instance.simplified_format(account) .details-counters - .counter{ class: active_nav_class(short_account_url(@account)) } - = link_to short_account_url(@account), class: 'u-url u-uid' do + .counter{ class: active_nav_class(short_account_url(account)) } + = link_to short_account_url(account), class: 'u-url u-uid' do %span.counter-label= t('accounts.posts') - %span.counter-number= number_with_delimiter @account.statuses_count - .counter{ class: active_nav_class(following_account_url(@account)) } - = link_to following_account_url(@account) do + %span.counter-number= number_with_delimiter account.statuses_count + .counter{ class: active_nav_class(following_account_url(account)) } + = link_to following_account_url(account) do %span.counter-label= t('accounts.following') - %span.counter-number= number_with_delimiter @account.following_count - .counter{ class: active_nav_class(followers_account_url(@account)) } - = link_to followers_account_url(@account) do + %span.counter-number= number_with_delimiter account.following_count + .counter{ class: active_nav_class(followers_account_url(account)) } + = link_to followers_account_url(account) do %span.counter-label= t('accounts.followers') - %span.counter-number= number_with_delimiter @account.followers_count + %span.counter-number= number_with_delimiter account.followers_count diff --git a/app/views/accounts/followers.html.haml b/app/views/accounts/followers.html.haml index fa5071f383be04f6d7a1d1381102c3fe8d355360..4b53aef0ccf33f50afb197e0e88ed372504a407b 100644 --- a/app/views/accounts/followers.html.haml +++ b/app/views/accounts/followers.html.haml @@ -1,7 +1,7 @@ - content_for :page_title do = t('accounts.people_who_follow', name: display_name(@account)) -= render partial: 'header' += render 'header', account: @account .accounts-grid - if @followers.empty? diff --git a/app/views/accounts/following.html.haml b/app/views/accounts/following.html.haml index 987dcba1fadf69c71efd1ad358496c9d01e91c01..4711997d9d43ec56d70013954efff0304e7624a1 100644 --- a/app/views/accounts/following.html.haml +++ b/app/views/accounts/following.html.haml @@ -1,7 +1,7 @@ - content_for :page_title do = t('accounts.people_followed_by', name: display_name(@account)) -= render partial: 'header' += render 'header', account: @account .accounts-grid - if @following.empty? diff --git a/app/views/accounts/show.html.haml b/app/views/accounts/show.html.haml index fd7ff9653d6f6c60797031f224b7cbd1e5ebc14f..9a70fd16fe05967d4945657467011546aec7b5db 100644 --- a/app/views/accounts/show.html.haml +++ b/app/views/accounts/show.html.haml @@ -20,7 +20,7 @@ .h-feed %data.p-name{ value: "#{@account.username} on #{Rails.configuration.x.local_domain}" }/ - = render partial: 'header' + = render 'header', account: @account - if @statuses.empty? .accounts-grid diff --git a/app/views/admin/domain_blocks/index.html.haml b/app/views/admin/domain_blocks/index.html.haml index 6f4ba9b5798d8e0f663098c92ef5e7dac596487c..da9a07bbc0b237dba3a399b60471e34f7b479128 100644 --- a/app/views/admin/domain_blocks/index.html.haml +++ b/app/views/admin/domain_blocks/index.html.haml @@ -6,12 +6,19 @@ %tr %th= t('admin.domain_block.domain') %th= t('admin.domain_block.severity') + %th= t('admin.domain_block.reject_media') + %th %tbody - @blocks.each do |block| %tr %td %samp= block.domain - %td= block.severity + %td= t("admin.domain_block.severities.#{block.severity}") + %td + - if block.reject_media? || block.suspend? + %i.fa.fa-check + %td + = table_link_to 'undo', t('admin.domain_block.undo'), admin_domain_block_path(block) = paginate @blocks = link_to t('admin.domain_block.add_new'), new_admin_domain_block_path, class: 'button' diff --git a/app/views/admin/domain_blocks/new.html.haml b/app/views/admin/domain_blocks/new.html.haml index 53aab21ff41c6ac3d3a6460d5979e6adc196519e..603faeb554b3b6a25ad4426ab758e1804675b0d8 100644 --- a/app/views/admin/domain_blocks/new.html.haml +++ b/app/views/admin/domain_blocks/new.html.haml @@ -10,5 +10,8 @@ = f.input :severity, collection: DomainBlock.severities.keys, wrapper: :with_label, include_blank: false, label_method: lambda { |type| I18n.t("admin.domain_block.new.severity.#{type}") } %p.hint= t('admin.domain_block.new.severity.desc_html') + + = f.input :reject_media, as: :boolean, wrapper: :with_label, label: I18n.t('admin.domain_block.reject_media'), hint: I18n.t('admin.domain_block.reject_media_hint') + .actions = f.button :button, t('admin.domain_block.new.create'), type: :submit diff --git a/app/views/admin/domain_blocks/show.html.haml b/app/views/admin/domain_blocks/show.html.haml new file mode 100644 index 0000000000000000000000000000000000000000..bf9011c5287ca4d311349c0f0e18eb763ce448b0 --- /dev/null +++ b/app/views/admin/domain_blocks/show.html.haml @@ -0,0 +1,9 @@ +- content_for :page_title do + = t('admin.domain_block.show.title', domain: @domain_block.domain) + += simple_form_for @domain_block, url: admin_domain_block_path(@domain_block), method: :delete do |f| + + = f.input :retroactive, as: :boolean, wrapper: :with_label, label: I18n.t("admin.domain_block.show.retroactive.#{@domain_block.severity}"), hint: I18n.t('admin.domain_block.show.affected_accounts', count: Account.where(domain: @domain_block.domain).count) + + .actions + = f.button :button, t('admin.domain_block.show.undo'), type: :submit diff --git a/app/views/api/v1/statuses/_media.rabl b/app/views/api/v1/statuses/_media.rabl index 2f56c6d07d82cf74a347fd03ba3fedff8736e1cb..80d80ea05a5c6b37f70c0c5440a94a08715fd785 100644 --- a/app/views/api/v1/statuses/_media.rabl +++ b/app/views/api/v1/statuses/_media.rabl @@ -1,5 +1,5 @@ attributes :id, :remote_url, :type -node(:url) { |media| full_asset_url(media.file.url(:original)) } -node(:preview_url) { |media| full_asset_url(media.file.url(:small)) } +node(:url) { |media| media.file.blank? ? media.remote_url : full_asset_url(media.file.url(:original)) } +node(:preview_url) { |media| media.file.blank? ? media.remote_url : full_asset_url(media.file.url(:small)) } node(:text_url) { |media| media.local? ? medium_url(media) : nil } diff --git a/app/views/settings/two_factor_auths/_recovery_codes.html.haml b/app/views/settings/two_factor_auths/_recovery_codes.html.haml index 719a1e01bad05ea9622ca43338feb21ab2bb00c6..054588b97d616e2bd4b00692017bc87c4702f0d9 100644 --- a/app/views/settings/two_factor_auths/_recovery_codes.html.haml +++ b/app/views/settings/two_factor_auths/_recovery_codes.html.haml @@ -1,6 +1,6 @@ %p.hint= t('two_factor_auth.recovery_instructions') %ol.recovery-codes - - @codes.each do |code| + - recovery_codes.each do |code| %li %samp= code diff --git a/app/views/settings/two_factor_auths/create.html.haml b/app/views/settings/two_factor_auths/create.html.haml index 8710b6e020f54ebf1999331ce27ff33ebfcb6115..138a930fd61b8da16965e089500e269d6d897902 100644 --- a/app/views/settings/two_factor_auths/create.html.haml +++ b/app/views/settings/two_factor_auths/create.html.haml @@ -1,4 +1,4 @@ - content_for :page_title do = t('settings.two_factor_auth') -= render 'recovery_codes' += render partial: 'recovery_codes', object: @codes diff --git a/app/views/settings/two_factor_auths/recovery_codes.html.haml b/app/views/settings/two_factor_auths/recovery_codes.html.haml index 8710b6e020f54ebf1999331ce27ff33ebfcb6115..138a930fd61b8da16965e089500e269d6d897902 100644 --- a/app/views/settings/two_factor_auths/recovery_codes.html.haml +++ b/app/views/settings/two_factor_auths/recovery_codes.html.haml @@ -1,4 +1,4 @@ - content_for :page_title do = t('settings.two_factor_auth') -= render 'recovery_codes' += render partial: 'recovery_codes', object: @codes diff --git a/app/views/stream_entries/_og_description.html.haml b/app/views/stream_entries/_og_description.html.haml new file mode 100644 index 0000000000000000000000000000000000000000..5762aca041cffb206d3e858ae85e990a41c5f3ae --- /dev/null +++ b/app/views/stream_entries/_og_description.html.haml @@ -0,0 +1,4 @@ +- if activity.is_a?(Status) && activity.spoiler_text? + %meta{ property: 'og:description', content: activity.spoiler_text }/ +- else + %meta{ property: 'og:description', content: activity.content }/ diff --git a/app/views/stream_entries/_og_image.html.haml b/app/views/stream_entries/_og_image.html.haml new file mode 100644 index 0000000000000000000000000000000000000000..f725209d82a1a490a88b044c09780e5f588106bf --- /dev/null +++ b/app/views/stream_entries/_og_image.html.haml @@ -0,0 +1,6 @@ +- if activity.is_a?(Status) && activity.non_sensitive_with_media? + %meta{ property: 'og:image', content: full_asset_url(activity.media_attachments.first.file.url(:small)) }/ +- else + %meta{ property: 'og:image', content: full_asset_url(account.avatar.url(:original)) }/ + %meta{ property: 'og:image:width', content: '120' }/ + %meta{ property: 'og:image:height', content: '120' }/ diff --git a/app/views/stream_entries/show.html.haml b/app/views/stream_entries/show.html.haml index eb8387ccb6c58910fb804a08bf9e7ac05d0ad079..dea5e9d4080e1bc0ece3bbfdc520f4ad585003e3 100644 --- a/app/views/stream_entries/show.html.haml +++ b/app/views/stream_entries/show.html.haml @@ -6,17 +6,8 @@ %meta{ property: 'og:type', content: 'article' }/ %meta{ property: 'og:title', content: "#{@account.username} on #{Rails.configuration.x.local_domain}" }/ - - if @stream_entry.activity.is_a?(Status) && !@stream_entry.activity.spoiler_text.blank? - %meta{ property: 'og:description', content: @stream_entry.activity.spoiler_text }/ - - else - %meta{ property: 'og:description', content: @stream_entry.activity.content }/ - - - if @stream_entry.activity.is_a?(Status) && !@stream_entry.activity.sensitive? && @stream_entry.activity.media_attachments.size > 0 - %meta{ property: 'og:image', content: full_asset_url(@stream_entry.activity.media_attachments.first.file.url(:small)) }/ - - else - %meta{ property: 'og:image', content: full_asset_url(@account.avatar.url(:original)) }/ - %meta{ property: 'og:image:width', content: '120' }/ - %meta{ property: 'og:image:height', content: '120' }/ + = render 'stream_entries/og_description', activity: @stream_entry.activity + = render 'stream_entries/og_image', activity: @stream_entry.activity, account: @account %meta{ property: 'twitter:card', content: 'summary' }/ diff --git a/bin/rspec b/bin/rspec new file mode 100755 index 0000000000000000000000000000000000000000..d738b23c0bab908c200b2a0ad83eba1fabd1c281 --- /dev/null +++ b/bin/rspec @@ -0,0 +1,17 @@ +#!/usr/bin/env ruby +# frozen_string_literal: true +# +# This file was generated by Bundler. +# +# The application 'rspec' is installed as part of a gem, and +# this file is here to facilitate running it. +# + +require "pathname" +ENV["BUNDLE_GEMFILE"] ||= File.expand_path("../../Gemfile", + Pathname.new(__FILE__).realpath) + +require "rubygems" +require "bundler/setup" + +load Gem.bin_path("rspec-core", "rspec") diff --git a/config/deploy.rb b/config/deploy.rb index 9bcf907be1ab375357aca558816d7976265e8f08..b1cade49d45223e91a6f80101ef1ef86c17e103d 100644 --- a/config/deploy.rb +++ b/config/deploy.rb @@ -1,4 +1,7 @@ -lock '3.7.2' +# frozen_string_literal: true + +lock '3.8.0' + set :repo_url, ENV.fetch('REPO', 'https://github.com/tootsuite/mastodon.git') set :branch, ENV.fetch('BRANCH', 'master') diff --git a/config/locales/en.yml b/config/locales/en.yml index 474de398533255891fdd4bf1a2712b730d439685..325df50450180a08e153364e11639079248032cf 100644 --- a/config/locales/en.yml +++ b/config/locales/en.yml @@ -81,6 +81,8 @@ en: web: Web domain_block: add_new: Add new + created_msg: Domain block is now being processed + destroyed_msg: Domain block has been undone domain: Domain new: create: Create block @@ -90,8 +92,22 @@ en: silence: Silence suspend: Suspend title: New domain block + reject_media: Reject media files + reject_media_hint: Removes locally stored media files and refuses to download any in the future. Irrelevant for suspensions + severities: + silence: Silence + suspend: Suspend severity: Severity + show: + affected_accounts: + one: One account in the database affected + other: "%{count} accounts in the database affected" + retroactive: + silence: Unsilence all existing accounts from this domain + suspend: Unsuspend all existing accounts from this domain + title: Undo domain block for %{domain} title: Domain Blocks + undo: Undo pubsubhubbub: callback_url: Callback URL confirmed: Confirmed diff --git a/config/locales/ja.yml b/config/locales/ja.yml index 3c7b342e2a021f2f4759449c12a005383bf171e2..5483e63b5e480cd46a2e21c864126bb228a45c2b 100644 --- a/config/locales/ja.yml +++ b/config/locales/ja.yml @@ -71,6 +71,7 @@ ja: profile_url: プãƒãƒ•ã‚£ãƒ¼ãƒ«URL public: パブリック push_subscription_expires: PuSHè³¼èªæœŸé™åˆ‡ã‚Œ + reset_password: パスワードå†è¨å®š salmon_url: Salmon URL silence: サイレンス statuses: トゥート数 @@ -81,6 +82,8 @@ ja: web: Web domain_block: add_new: æ–°è¦è¿½åŠ + created_msg: ドメインブãƒãƒƒã‚¯å‡¦ç†ã‚’完了ã—ã¾ã—㟠+ destroyed_msg: ドメインブãƒãƒƒã‚¯ã‚’外ã—ã¾ã—㟠domain: ドメイン new: create: ブãƒãƒƒã‚¯ã‚’ä½œæˆ @@ -90,8 +93,21 @@ ja: silence: サイレンス suspend: åœæ¢ title: æ–°è¦ãƒ‰ãƒ¡ã‚¤ãƒ³ãƒ–ãƒãƒƒã‚¯ + reject_media: ãƒ¡ãƒ‡ã‚£ã‚¢ãƒ•ã‚¡ã‚¤ãƒ«ã‚’æ‹’å¦ + reject_media_hint: ãƒãƒ¼ã‚«ãƒ«ã«ä¿æ‘ã•ã‚ŒãŸãƒ¡ãƒ‡ã‚£ã‚¢ãƒ•ã‚¡ã‚¤ãƒ«ã‚’削除ã—ã€ä»Šå¾Œã®ãƒ€ã‚¦ãƒ³ãƒãƒ¼ãƒ‰ã‚’æ‹’å¦ã—ã¾ã™ã€‚åœæ¢ã¨ã¯ç„¡é–¢ä¿‚ã§ã™ã€‚ + severities: + silence: サイレンス + suspend: åœæ¢ severity: 深刻度 + show: + affected_accounts: "データベースä¸ã®%{count}個ã®ã‚¢ã‚«ã‚¦ãƒ³ãƒˆã«å½±éŸ¿ã—ã¾ã™" + retroactive: + silence: ã“ã®ãƒ‰ãƒ¡ã‚¤ãƒ³ã‹ã‚‰ã®å˜åœ¨ã™ã‚‹ã™ã¹ã¦ã®ã‚¢ã‚«ã‚¦ãƒ³ãƒˆã®ã‚µã‚¤ãƒ¬ãƒ³ã‚¹ã‚’戻㙠+ suspend: ã“ã®ãƒ‰ãƒ¡ã‚¤ãƒ³ã‹ã‚‰ã®å˜åœ¨ã™ã‚‹ã™ã¹ã¦ã®ã‚¢ã‚«ã‚¦ãƒ³ãƒˆã®åœæ¢ã‚’戻㙠+ title: "%{domain}ã®ãƒ‰ãƒ¡ã‚¤ãƒ³ãƒ–ãƒãƒƒã‚¯ã‚’戻ã™" + undo: å…ƒã«æˆ»ã™ title: ドメインブãƒãƒƒã‚¯ + undo: å…ƒã«æˆ»ã™ pubsubhubbub: callback_url: コールãƒãƒƒã‚¯URL confirmed: 確èªæ¸ˆã¿ @@ -106,7 +122,7 @@ ja: delete: 削除 id: ID mark_as_resolved: 解決済ã¿ã¨ã—ã¦ãƒžãƒ¼ã‚¯ - report: 'レãƒãƒ¼ãƒˆ#%{id}' + report: レãƒãƒ¼ãƒˆ#%{id} reported_account: å ±å‘Šå¯¾è±¡ã‚¢ã‚«ã‚¦ãƒ³ãƒˆ reported_by: å ±å‘Šè€… resolved: 解決済㿠@@ -290,8 +306,13 @@ ja: disable: 無効 enable: 有効 enabled_success: 二段階èªè¨¼ãŒæœ‰åŠ¹ã«ãªã‚Šã¾ã—㟠+ generate_recovery_codes: å¾©å…ƒã‚³ãƒ¼ãƒ‰ã‚’ç”Ÿæˆ instructions_html: "<strong>Google Authenticatorã‹ã€ã‚‚ã—ãã¯ã»ã‹ã®TOTPアプリã§ã“ã®QRコードをスã‚ャンã—ã¦ãã ã•ã„。</strong>ã“れ以é™ã€ãƒã‚°ã‚¤ãƒ³ã™ã‚‹ã¨ãã¯ãã®ã‚¢ãƒ—リã§ç”Ÿæˆã•ã‚Œã‚‹ã‚³ãƒ¼ãƒ‰ãŒå¿…è¦ã«ãªã‚Šã¾ã™ã€‚" + lost_recovery_codes: リカãƒãƒªã‚³ãƒ¼ãƒ‰ã‚’使用ã™ã‚‹ã¨æºå¸¯é›»è©±ã‚’紛失ã—ãŸå ´åˆã§ã‚‚アカウントã«ã‚¢ã‚¯ã‚»ã‚¹ã§ãるよã†ã«ãªã‚Šã¾ã™ã€‚ リカãƒãƒªãƒ¼ã‚³ãƒ¼ãƒ‰ã‚’紛失ã—ãŸå ´åˆã‚‚ã“ã“ã§å†ç”Ÿæˆã™ã‚‹ã“ã¨ãŒã§ãã¾ã™ãŒã€å¤ã„リカãƒãƒªã‚³ãƒ¼ãƒ‰ã¯ç„¡åŠ¹ã«ãªã‚Šã¾ã™ã€‚ manual_instructions: 'QRコードãŒã‚¹ã‚ャンã§ããšã€æ‰‹å‹•ã§ã®ç™»éŒ²ã‚’希望ã®å ´åˆã¯ã“ã®ã‚·ãƒ¼ã‚¯ãƒ¬ãƒƒãƒˆã‚³ãƒ¼ãƒ‰ã‚’利用ã—ã¦ãã ã•ã„。:' + recovery_codes: リカãƒãƒªãƒ¼ã‚³ãƒ¼ãƒ‰ + recovery_codes_regenerated: リカãƒãƒªãƒ¼ã‚³ãƒ¼ãƒ‰ãŒå†ç”Ÿæˆã•ã‚Œã¾ã—ãŸã€‚ + recovery_instructions: æºå¸¯é›»è©±ã‚’紛失ã—ãŸå ´åˆã€ä»¥ä¸‹ã®å†…ã©ã‚Œã‹ã®ãƒªã‚«ãƒãƒªã‚³ãƒ¼ãƒ‰ã‚’使用ã—ã¦ã‚¢ã‚«ã‚¦ãƒ³ãƒˆã¸ã‚¢ã‚¯ã‚»ã‚¹ã™ã‚‹ã“ã¨ãŒã§ãã¾ã™ã€‚ リカãƒãƒªã‚³ãƒ¼ãƒ‰ã¯å°åˆ·ã—ã¦å®‰å…¨ã«ä¿ç®¡ã—ã¦ãã ã•ã„。 setup: åˆæœŸè¨å®š warning: ç¾åœ¨èªè¨¼ã‚¢ãƒ—リをè¨å®šã§ããªã„å ´åˆã€ç„¡åŠ¹ã«è¨å®šã—ã¦ã€æœ‰åŠ¹ã«ã—ãªã„ã§ãã ã•ã„。 wrong_code: コードãŒé–“é•ã£ã¦ã„ã¾ã™ã€‚サーãƒãƒ¼ä¸Šã®æ™‚é–“ã¨ãƒ‡ãƒã‚¤ã‚¹ä¸Šã®æ™‚é–“ãŒä¸€è‡´ã—ã¦ã„ã‚‹ã“ã¨ã‚’確èªã—ã¦ãã ã•ã„。 diff --git a/config/locales/pt.yml b/config/locales/pt.yml index f2c7458f7f2df351c70d0675277b19a3006c9afc..703c8467fbcbcc5b4c9e56d38f4dc8a8806aa0cc 100644 --- a/config/locales/pt.yml +++ b/config/locales/pt.yml @@ -2,28 +2,97 @@ pt: about: about_mastodon: Mastodon é um servidor de rede social <em>grátis, e open-source</em>. Uma alternativa <em>descentralizada</em> ás plataformas comerciais, que evita o risco de uma única empresa monopolizar a sua comunicação. Qualquer um pode ter uma instância Mastodon e assim participar na <em>rede social federada</em> sem problemas. + about_this: Sobre essa instância get_started: Como começar + apps: Aplicações + business_email: 'Email comercial:' + closed_registrations: Registros estão fechadas para essa instância. + contact: Contato + description_headline: O que é %{domain}? + domain_count_after: outras instâncias + domain_count_before: Conectado a + features: + api: Aberto para API de aplicações e serviços + blocks: Bloqueos e ferramentas para mudar + characters: 500 caracteres por post + chronology: Timeline são cronologicas + ethics: 'Design ético: sem propaganda, sem tracking' + gifv: GIFV e vÃdeos curtos + privacy: Granular, privacidade setada por post + public: Timelines públicas + features_headline: O que torna Mastodon diferente + get_started: Comece aqui + links: Links source_code: Source code + other_instances: Outras instâncias terms: Termos + user_count_after: usuários + user_count_before: Lugar de accounts: follow: Seguir followers: Seguidores - following: Following + following: Seguindo nothing_here: Não há nada aqui! people_followed_by: Pessoas seguidas por %{name} people_who_follow: Pessoas que seguem %{name} posts: Posts + remote_follow: Acesso remoto unfollow: Unfollow + admin: + accounts: + are_you_sure: Você tem certeza? + display_name: Nome mostrado + domain: Domain + edit: Editar + email: E-mail + feed_url: URL do Feed + followers: Seguidores + follows: Seguindo + location: + all: Todos + local: Local + remote: Remoto + title: Local + media_attachments: MÃdia anexadas + moderation: + all: Todos + silenced: Silenciado + suspended: Supenso + title: Moderação + most_recent_activity: Atividade mais recente + most_recent_ip: IP mais recente + not_subscribed: Não inscrito + order: + alphabetic: Alfabética + most_recent: Mais recente + title: Ordem + perform_full_suspension: Fazer suspensão completa + profile_url: URL do perfil + public: Público + push_subscription_expires: PuSH subscription expires + salmon_url: Salmon URL + silence: Silêncio + statuses: Status + title: Contas + undo_silenced: Desfazer silenciar + undo_suspension: Desfazer supensão + username: Usuário + web: Web + domain_block: + add_new: Adicionar nova + created_msg: Bloqueio do domÃnio está sendo processado + destroyed_msg: Bloqueio de domÃnio está sendo desfeito + domain: DomÃnio application_mailer: signature: notificações Mastodon de %{instance} auth: - change_password: Mudar password + change_password: Mudar senha didnt_get_confirmation: Não recebeu instruções de confirmação? - forgot_password: Esqueceu a password? + forgot_password: Esqueceu a senha? login: Entrar register: Registar resend_confirmation: Reenviar instruções de confirmação - reset_password: Reset password + reset_password: Resetar senha set_new_password: Editar password generic: changes_saved_msg: Mudanças guardadas! diff --git a/config/locales/simple_form.ja.yml b/config/locales/simple_form.ja.yml index 103001b7efc3917325e1ec051963fc85f0b242eb..b9257ec35fe9fbe14b876d018c3a7b3ef23f6d21 100644 --- a/config/locales/simple_form.ja.yml +++ b/config/locales/simple_form.ja.yml @@ -10,11 +10,13 @@ ja: note: プãƒãƒ•ã‚£ãƒ¼ãƒ«ã¯160æ–‡å—ã¾ã§è¨å®šã™ã‚‹ã“ã¨ãŒã§ãã¾ã™ã€‚ imports: data: ä»–ã® Mastodon サーãƒãƒ¼ã‹ã‚‰ã‚¨ã‚¯ã‚¹ãƒãƒ¼ãƒˆã—ãŸCSVファイルをé¸æŠžã—ã¦ä¸‹ã•ã„ + sessions: + otp: æºå¸¯é›»è©±ã«è¡¨ç¤ºã•ã‚ŒãŸ2段階èªè¨¼ã‚³ãƒ¼ãƒ‰ã‚’入力ã™ã‚‹ã‹ã€ç”Ÿæˆã—ãŸãƒªã‚«ãƒãƒªãƒ¼ã‚³ãƒ¼ãƒ‰ã‚’使用ã—ã¦ãã ã•ã„。 labels: defaults: avatar: アイコン confirm_new_password: æ–°ã—ã„パスワード(確èªç”¨ï¼‰ - confirm_password: æ–°ã—ã„パスワード + confirm_password: パスワード(確èªç”¨ï¼‰ current_password: ç¾åœ¨ã®ãƒ‘スワード data: データ display_name: 表示å @@ -22,12 +24,13 @@ ja: header: ヘッダー locale: 言語 locked: éžå…¬é–‹ã‚¢ã‚«ã‚¦ãƒ³ãƒˆã«ã™ã‚‹ - new_password: パスワード + new_password: æ–°ã—ã„パスワード note: プãƒãƒ•ã‚£ãƒ¼ãƒ« otp_attempt: 二段階èªè¨¼ã‚³ãƒ¼ãƒ‰ password: パスワード setting_boost_modal: ブーストã™ã‚‹å‰ã«ç¢ºèªãƒ€ã‚¤ã‚¢ãƒã‚°ã‚’表示ã™ã‚‹ setting_default_privacy: 投稿ã®å…¬é–‹ç¯„囲 + severity: é‡å¤§æ€§ type: インãƒãƒ¼ãƒˆã™ã‚‹é …ç›® username: ユーザーå interactions: diff --git a/config/locales/simple_form.pt.yml b/config/locales/simple_form.pt.yml index 07099c1f0ddde00ec026fc0de2c670cf113a564a..e8b5e2d7f06a89d50e81645c3781d3e5594e487d 100644 --- a/config/locales/simple_form.pt.yml +++ b/config/locales/simple_form.pt.yml @@ -4,17 +4,17 @@ pt: labels: defaults: avatar: Avatar - confirm_new_password: Confirme nova password - confirm_password: Confirme a password - current_password: Password atual + confirm_new_password: Confirme nova senha + confirm_password: Confirme a senha + current_password: Senha atual display_name: Nome email: Endereço de email header: Header locale: Linguagem - new_password: Nova password + new_password: Nova senha note: Biografia - password: Password - username: Username + password: Senha + username: Usuário interactions: must_be_follower: Bloquear notificações de não-seguidores must_be_following: Bloquear notificações de pessoas que você diff --git a/config/routes.rb b/config/routes.rb index 31909a4f4d167f7907ed9223a5523f221632af01..fd186c3206a57e821a68f22b3a02d068bd3efd5c 100644 --- a/config/routes.rb +++ b/config/routes.rb @@ -78,7 +78,7 @@ Rails.application.routes.draw do namespace :admin do resources :pubsubhubbub, only: [:index] - resources :domain_blocks, only: [:index, :new, :create] + resources :domain_blocks, only: [:index, :new, :create, :show, :destroy] resources :settings, only: [:index, :update] resources :reports, only: [:index, :show, :update] do diff --git a/lib/tasks/mastodon.rake b/lib/tasks/mastodon.rake index 54980634d354af27e92959bf91c4485063a23acd..b477302740175722be9cf2eb221282344b218e7f 100644 --- a/lib/tasks/mastodon.rake +++ b/lib/tasks/mastodon.rake @@ -1,6 +1,16 @@ # frozen_string_literal: true namespace :mastodon do + desc 'Execute daily tasks' + task :daily do + Rake::Task['mastodon:feeds:clear'].invoke + Rake::Task['mastodon:media:clear'].invoke + Rake::Task['mastodon:users:clear'].invoke + + Rake::Task['mastodon:push:refresh'].invoke + end + + desc 'Turn a user into an admin, identified by the USERNAME environment variable' task make_admin: :environment do include RoutingHelper @@ -13,12 +23,13 @@ namespace :mastodon do desc 'Manually confirms a user with associated user email address stored in USER_EMAIL environment variable.' task confirm_email: :environment do email = ENV.fetch('USER_EMAIL') - user = User.where(email: email).first + user = User.find_by(email: email) + if user user.update(confirmed_at: Time.now.utc) - puts "User #{email} confirmed." + puts "#{email} confirmed" else - abort "User #{email} not found." + abort "#{email} not found" end end @@ -32,6 +43,13 @@ namespace :mastodon do task remove_silenced: :environment do MediaAttachment.where(account: Account.silenced).find_each(&:destroy) end + + desc 'Remove cached remote media attachments that are older than a week' + task remove_remote: :environment do + MediaAttachment.where.not(remote_url: '').where('created_at < ?', 1.week.ago).find_each do |media| + media.file.destroy + end + end end namespace :push do @@ -60,7 +78,7 @@ namespace :mastodon do end end - desc 'Clears all timelines so that they would be regenerated on next hit' + desc 'Clears all timelines' task clear_all: :environment do Redis.current.keys('feed:*').each { |key| Redis.current.del(key) } end @@ -126,8 +144,13 @@ namespace :mastodon do Rails.logger.debug 'Generating static avatars/headers for GIF ones...' Account.unscoped.where(avatar_content_type: 'image/gif').or(Account.unscoped.where(header_content_type: 'image/gif')).find_each do |account| - account.avatar.reprocess! - account.header.reprocess! + begin + account.avatar.reprocess! + account.header.reprocess! + rescue StandardError => e + Rails.logger.error "Error while generating static avatars/headers for account #{account.id}: #{e}" + next + end end Rails.logger.debug 'Done!' diff --git a/package.json b/package.json index 6f448238bd8893fe8174874ff2a1305e6015a925..0ced631a9cb9f30b9ddf43dc6b101c32b17e8bbb 100644 --- a/package.json +++ b/package.json @@ -1,5 +1,6 @@ { "name": "mastodon", + "license" : "AGPL-3.0", "scripts": { "start": "babel-node ./streaming/index.js --presets es2015,stage-2", "storybook": "start-storybook -p 9001 -c storybook", diff --git a/spec/views/stream_entries/show.html.haml_spec.rb b/spec/views/stream_entries/show.html.haml_spec.rb index 5526d6780ed70f2c1dc9f277a85454761e9146d6..acc0c39f5b11b11fe3bf8f2771918cfb9c903b65 100644 --- a/spec/views/stream_entries/show.html.haml_spec.rb +++ b/spec/views/stream_entries/show.html.haml_spec.rb @@ -61,5 +61,4 @@ describe 'stream_entries/show.html.haml' do expect(mf2.entry.in_reply_to.format.author.format.name.to_s).to eq alice.display_name expect(mf2.entry.in_reply_to.format.author.format.url.to_s).not_to be_empty end - end