Make file attachment on MediaAttachment optional (#1865)
Create MediaAttachment but without actual file download when domain is blocked with reject_media set to true Clean up old media files when creating a new domain block with reject_media set to true Return remote_url in media attachments API if local file is not present Undo domain block action in admin UI Ability to enable reject_media from admin UIgh/stable
parent
8a58942c80
commit
5d710b1139
|
@ -15,16 +15,26 @@ module Admin
|
||||||
|
|
||||||
if @domain_block.save
|
if @domain_block.save
|
||||||
DomainBlockWorker.perform_async(@domain_block.id)
|
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
|
else
|
||||||
render action: :new
|
render action: :new
|
||||||
end
|
end
|
||||||
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
|
private
|
||||||
|
|
||||||
def resource_params
|
def resource_params
|
||||||
params.require(:domain_block).permit(:domain, :severity)
|
params.require(:domain_block).permit(:domain, :severity, :reject_media, :retroactive)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -8,7 +8,9 @@ class ApplicationController < ActionController::Base
|
||||||
force_ssl if: "Rails.env.production? && ENV['LOCAL_HTTPS'] == 'true'"
|
force_ssl if: "Rails.env.production? && ENV['LOCAL_HTTPS'] == 'true'"
|
||||||
|
|
||||||
include Localized
|
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 ActionController::RoutingError, with: :not_found
|
||||||
rescue_from ActiveRecord::RecordNotFound, with: :not_found
|
rescue_from ActiveRecord::RecordNotFound, with: :not_found
|
||||||
|
|
|
@ -3,6 +3,8 @@
|
||||||
class DomainBlock < ApplicationRecord
|
class DomainBlock < ApplicationRecord
|
||||||
enum severity: [:silence, :suspend]
|
enum severity: [:silence, :suspend]
|
||||||
|
|
||||||
|
attr_accessor :retroactive
|
||||||
|
|
||||||
validates :domain, presence: true, uniqueness: true
|
validates :domain, presence: true, uniqueness: true
|
||||||
|
|
||||||
def self.blocked?(domain)
|
def self.blocked?(domain)
|
||||||
|
|
|
@ -3,12 +3,34 @@
|
||||||
class BlockDomainService < BaseService
|
class BlockDomainService < BaseService
|
||||||
def call(domain_block)
|
def call(domain_block)
|
||||||
if domain_block.silence?
|
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
|
else
|
||||||
Account.where(domain: domain_block.domain).find_each do |account|
|
suspend_accounts!(domain_block.domain)
|
||||||
account.subscription(api_subscription_url(account.id)).unsubscribe if account.subscribed?
|
end
|
||||||
SuspendAccountService.new.call(account)
|
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
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -179,12 +179,12 @@ class ProcessFeedService < BaseService
|
||||||
end
|
end
|
||||||
|
|
||||||
def hashtags_from_xml(parent, xml)
|
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)
|
ProcessHashtagsService.new.call(parent, tags)
|
||||||
end
|
end
|
||||||
|
|
||||||
def media_from_xml(parent, xml)
|
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|
|
xml.xpath('./xmlns:link[@rel="enclosure"]', xmlns: TagManager::XMLNS).each do |link|
|
||||||
next unless link['href']
|
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'])
|
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'])
|
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
|
begin
|
||||||
media.file_remote_url = link['href']
|
media.file_remote_url = link['href']
|
||||||
|
|
|
@ -13,6 +13,7 @@ class SuspendAccountService < BaseService
|
||||||
|
|
||||||
def purge_content
|
def purge_content
|
||||||
@account.statuses.reorder(nil).find_each do |status|
|
@account.statuses.reorder(nil).find_each do |status|
|
||||||
|
# This federates out deletes to previous followers
|
||||||
RemoveStatusService.new.call(status)
|
RemoveStatusService.new.call(status)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -29,9 +30,7 @@ class SuspendAccountService < BaseService
|
||||||
@account.display_name = ''
|
@account.display_name = ''
|
||||||
@account.note = ''
|
@account.note = ''
|
||||||
@account.avatar.destroy
|
@account.avatar.destroy
|
||||||
@account.avatar.clear
|
|
||||||
@account.header.destroy
|
@account.header.destroy
|
||||||
@account.header.clear
|
|
||||||
@account.save!
|
@account.save!
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
|
@ -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
|
|
@ -6,12 +6,19 @@
|
||||||
%tr
|
%tr
|
||||||
%th= t('admin.domain_block.domain')
|
%th= t('admin.domain_block.domain')
|
||||||
%th= t('admin.domain_block.severity')
|
%th= t('admin.domain_block.severity')
|
||||||
|
%th= t('admin.domain_block.reject_media')
|
||||||
|
%th
|
||||||
%tbody
|
%tbody
|
||||||
- @blocks.each do |block|
|
- @blocks.each do |block|
|
||||||
%tr
|
%tr
|
||||||
%td
|
%td
|
||||||
%samp= block.domain
|
%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
|
= paginate @blocks
|
||||||
= link_to t('admin.domain_block.add_new'), new_admin_domain_block_path, class: 'button'
|
= link_to t('admin.domain_block.add_new'), new_admin_domain_block_path, class: 'button'
|
||||||
|
|
|
@ -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}") }
|
= 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')
|
%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
|
.actions
|
||||||
= f.button :button, t('admin.domain_block.new.create'), type: :submit
|
= f.button :button, t('admin.domain_block.new.create'), type: :submit
|
||||||
|
|
|
@ -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
|
|
@ -1,5 +1,5 @@
|
||||||
attributes :id, :remote_url, :type
|
attributes :id, :remote_url, :type
|
||||||
|
|
||||||
node(:url) { |media| full_asset_url(media.file.url(:original)) }
|
node(:url) { |media| media.file.blank? ? media.remote_url : full_asset_url(media.file.url(:original)) }
|
||||||
node(:preview_url) { |media| full_asset_url(media.file.url(:small)) }
|
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 }
|
node(:text_url) { |media| media.local? ? medium_url(media) : nil }
|
||||||
|
|
|
@ -81,6 +81,8 @@ en:
|
||||||
web: Web
|
web: Web
|
||||||
domain_block:
|
domain_block:
|
||||||
add_new: Add new
|
add_new: Add new
|
||||||
|
created_msg: Domain block is now being processed
|
||||||
|
destroyed_msg: Domain block has been undone
|
||||||
domain: Domain
|
domain: Domain
|
||||||
new:
|
new:
|
||||||
create: Create block
|
create: Create block
|
||||||
|
@ -90,8 +92,22 @@ en:
|
||||||
silence: Silence
|
silence: Silence
|
||||||
suspend: Suspend
|
suspend: Suspend
|
||||||
title: New domain block
|
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
|
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
|
title: Domain Blocks
|
||||||
|
undo: Undo
|
||||||
pubsubhubbub:
|
pubsubhubbub:
|
||||||
callback_url: Callback URL
|
callback_url: Callback URL
|
||||||
confirmed: Confirmed
|
confirmed: Confirmed
|
||||||
|
|
|
@ -78,7 +78,7 @@ Rails.application.routes.draw do
|
||||||
|
|
||||||
namespace :admin do
|
namespace :admin do
|
||||||
resources :pubsubhubbub, only: [:index]
|
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 :settings, only: [:index, :update]
|
||||||
|
|
||||||
resources :reports, only: [:index, :show, :update] do
|
resources :reports, only: [:index, :show, :update] do
|
||||||
|
|
|
@ -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.name.to_s).to eq alice.display_name
|
||||||
expect(mf2.entry.in_reply_to.format.author.format.url.to_s).not_to be_empty
|
expect(mf2.entry.in_reply_to.format.author.format.url.to_s).not_to be_empty
|
||||||
end
|
end
|
||||||
|
|
||||||
end
|
end
|
||||||
|
|
Reference in New Issue