mirror of
https://github.com/mastodon/mastodon.git
synced 2025-01-18 16:11:42 +01:00
cbd0ee1d07
* Update devise-two-factor to unreleased fork for Rails 6 support Update tests to match new `rotp` version. * Update nsa gem to unreleased fork for Rails 6 support * Update rails to 6.1.3 and rails-i18n to 6.0 * Update to unreleased fork of pluck_each for Ruby 6 support * Run "rails app:update" * Add missing ActiveStorage config file * Use config.ssl_options instead of removed ApplicationController#force_ssl Disabled force_ssl-related tests as they do not seem to be easily testable anymore. * Fix nonce directives by removing Rails 5 specific monkey-patching * Fix fixture_file_upload deprecation warning * Fix yield-based test failing with Rails 6 * Use Rails 6's index_with when possible * Use ActiveRecord::Cache::Store#delete_multi from Rails 6 This will yield better performances when deleting an account * Disable Rails 6.1's automatic preload link headers Since Rails 6.1, ActionView adds preload links for javascript files in the Links header per default. In our case, that will bloat headers too much and potentially cause issues with reverse proxies. Furhermore, we don't need those links, as we already output them as HTML link tags. * Switch to Rails 6.0 default config * Switch to Rails 6.1 default config * Do not include autoload paths in the load path
147 lines
4.5 KiB
Ruby
147 lines
4.5 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
require 'csv'
|
|
|
|
class ImportService < BaseService
|
|
ROWS_PROCESSING_LIMIT = 20_000
|
|
|
|
def call(import)
|
|
@import = import
|
|
@account = @import.account
|
|
|
|
case @import.type
|
|
when 'following'
|
|
import_follows!
|
|
when 'blocking'
|
|
import_blocks!
|
|
when 'muting'
|
|
import_mutes!
|
|
when 'domain_blocking'
|
|
import_domain_blocks!
|
|
when 'bookmarks'
|
|
import_bookmarks!
|
|
end
|
|
end
|
|
|
|
private
|
|
|
|
def import_follows!
|
|
parse_import_data!(['Account address'])
|
|
import_relationships!('follow', 'unfollow', @account.following, ROWS_PROCESSING_LIMIT, reblogs: { header: 'Show boosts', default: true })
|
|
end
|
|
|
|
def import_blocks!
|
|
parse_import_data!(['Account address'])
|
|
import_relationships!('block', 'unblock', @account.blocking, ROWS_PROCESSING_LIMIT)
|
|
end
|
|
|
|
def import_mutes!
|
|
parse_import_data!(['Account address'])
|
|
import_relationships!('mute', 'unmute', @account.muting, ROWS_PROCESSING_LIMIT, notifications: { header: 'Hide notifications', default: true })
|
|
end
|
|
|
|
def import_domain_blocks!
|
|
parse_import_data!(['#domain'])
|
|
items = @data.take(ROWS_PROCESSING_LIMIT).map { |row| row['#domain'].strip }
|
|
|
|
if @import.overwrite?
|
|
presence_hash = items.index_with(true)
|
|
|
|
@account.domain_blocks.find_each do |domain_block|
|
|
if presence_hash[domain_block.domain]
|
|
items.delete(domain_block.domain)
|
|
else
|
|
@account.unblock_domain!(domain_block.domain)
|
|
end
|
|
end
|
|
end
|
|
|
|
items.each do |domain|
|
|
@account.block_domain!(domain)
|
|
end
|
|
|
|
AfterAccountDomainBlockWorker.push_bulk(items) do |domain|
|
|
[@account.id, domain]
|
|
end
|
|
end
|
|
|
|
def import_relationships!(action, undo_action, overwrite_scope, limit, extra_fields = {})
|
|
local_domain_suffix = "@#{Rails.configuration.x.local_domain}"
|
|
items = @data.take(limit).map { |row| [row['Account address']&.strip&.delete_suffix(local_domain_suffix), Hash[extra_fields.map { |key, field_settings| [key, row[field_settings[:header]]&.strip || field_settings[:default]] }]] }.reject { |(id, _)| id.blank? }
|
|
|
|
if @import.overwrite?
|
|
presence_hash = items.each_with_object({}) { |(id, extra), mapping| mapping[id] = [true, extra] }
|
|
|
|
overwrite_scope.find_each do |target_account|
|
|
if presence_hash[target_account.acct]
|
|
items.delete(target_account.acct)
|
|
extra = presence_hash[target_account.acct][1]
|
|
Import::RelationshipWorker.perform_async(@account.id, target_account.acct, action, extra)
|
|
else
|
|
Import::RelationshipWorker.perform_async(@account.id, target_account.acct, undo_action)
|
|
end
|
|
end
|
|
end
|
|
|
|
head_items = items.uniq { |acct, _| acct.split('@')[1] }
|
|
tail_items = items - head_items
|
|
|
|
Import::RelationshipWorker.push_bulk(head_items + tail_items) do |acct, extra|
|
|
[@account.id, acct, action, extra]
|
|
end
|
|
end
|
|
|
|
def import_bookmarks!
|
|
parse_import_data!(['#uri'])
|
|
items = @data.take(ROWS_PROCESSING_LIMIT).map { |row| row['#uri'].strip }
|
|
|
|
if @import.overwrite?
|
|
presence_hash = items.index_with(true)
|
|
|
|
@account.bookmarks.find_each do |bookmark|
|
|
if presence_hash[bookmark.status.uri]
|
|
items.delete(bookmark.status.uri)
|
|
else
|
|
bookmark.destroy!
|
|
end
|
|
end
|
|
end
|
|
|
|
statuses = items.filter_map do |uri|
|
|
status = ActivityPub::TagManager.instance.uri_to_resource(uri, Status)
|
|
next if status.nil? && ActivityPub::TagManager.instance.local_uri?(uri)
|
|
|
|
status || ActivityPub::FetchRemoteStatusService.new.call(uri)
|
|
end
|
|
|
|
account_ids = statuses.map(&:account_id)
|
|
preloaded_relations = relations_map_for_account(@account, account_ids)
|
|
|
|
statuses.keep_if { |status| StatusPolicy.new(@account, status, preloaded_relations).show? }
|
|
|
|
statuses.each do |status|
|
|
@account.bookmarks.find_or_create_by!(account: @account, status: status)
|
|
end
|
|
end
|
|
|
|
def parse_import_data!(default_headers)
|
|
data = CSV.parse(import_data, headers: true)
|
|
data = CSV.parse(import_data, headers: default_headers) unless data.headers&.first&.strip&.include?(' ')
|
|
@data = data.reject(&:blank?)
|
|
end
|
|
|
|
def import_data
|
|
Paperclip.io_adapters.for(@import.data).read
|
|
end
|
|
|
|
def relations_map_for_account(account, account_ids)
|
|
{
|
|
blocking: {},
|
|
blocked_by: Account.blocked_by_map(account_ids, account.id),
|
|
muting: {},
|
|
following: Account.following_map(account_ids, account.id),
|
|
domain_blocking_by_domain: {},
|
|
}
|
|
end
|
|
end
|