You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
118 lines
4.3 KiB
118 lines
4.3 KiB
# frozen_string_literal: true |
|
|
|
require_relative '../../lib/mastodon/migration_warning' |
|
|
|
class FixAccountsUniqueIndex < ActiveRecord::Migration[5.2] |
|
include Mastodon::MigrationWarning |
|
|
|
class Account < ApplicationRecord |
|
# Dummy class, to make migration possible across version changes |
|
has_one :user, inverse_of: :account |
|
|
|
def local? |
|
domain.nil? |
|
end |
|
|
|
def acct |
|
local? ? username : "#{username}@#{domain}" |
|
end |
|
end |
|
|
|
class StreamEntry < ApplicationRecord |
|
# Dummy class, to make migration possible across version changes |
|
belongs_to :account, inverse_of: :stream_entries |
|
end |
|
|
|
class Status < ApplicationRecord |
|
# Dummy class, to make migration possible across version changes |
|
belongs_to :account |
|
end |
|
|
|
class Mention < ApplicationRecord |
|
# Dummy class, to make migration possible across version changes |
|
belongs_to :account |
|
end |
|
|
|
class StatusPin < ApplicationRecord |
|
# Dummy class, to make migration possible across version changes |
|
belongs_to :account |
|
end |
|
|
|
disable_ddl_transaction! |
|
|
|
def up |
|
migration_duration_warning(<<~EXPLANATION) |
|
This migration will irreversibly delete user accounts with duplicate |
|
usernames. You may use the `rake mastodon:maintenance:find_duplicate_usernames` |
|
task to manually deal with such accounts before running this migration. |
|
EXPLANATION |
|
|
|
duplicates = Account.connection.select_all('SELECT string_agg(id::text, \',\') AS ids FROM accounts GROUP BY lower(username), lower(domain) HAVING count(*) > 1').to_ary |
|
|
|
duplicates.each do |row| |
|
deduplicate_account!(row['ids'].split(',')) |
|
end |
|
|
|
remove_index :accounts, name: 'index_accounts_on_username_and_domain_lower' if index_name_exists?(:accounts, 'index_accounts_on_username_and_domain_lower') |
|
safety_assured { execute 'CREATE UNIQUE INDEX CONCURRENTLY index_accounts_on_username_and_domain_lower ON accounts (lower(username), lower(domain))' } |
|
remove_index :accounts, name: 'index_accounts_on_username_and_domain' if index_name_exists?(:accounts, 'index_accounts_on_username_and_domain') |
|
end |
|
|
|
def down |
|
raise ActiveRecord::IrreversibleMigration |
|
end |
|
|
|
private |
|
|
|
def deduplicate_account!(account_ids) |
|
accounts = Account.where(id: account_ids).to_a |
|
accounts = accounts.first.local? ? accounts.sort_by(&:created_at) : accounts.sort_by(&:updated_at).reverse |
|
reference_account = accounts.shift |
|
|
|
say_with_time "Deduplicating @#{reference_account.acct} (#{accounts.size} duplicates)..." do |
|
accounts.each do |other_account| |
|
if other_account.public_key == reference_account.public_key |
|
# The accounts definitely point to the same resource, so |
|
# it's safe to re-attribute content and relationships |
|
merge_accounts!(reference_account, other_account) |
|
elsif other_account.local? |
|
# Since domain is in the GROUP BY clause, both accounts |
|
# are always either going to be local or not local, so only |
|
# one check is needed. Since we cannot support two users with |
|
# the same username locally, one has to go. 😢 |
|
other_account.user&.destroy |
|
end |
|
|
|
other_account.destroy |
|
end |
|
end |
|
end |
|
|
|
def merge_accounts!(main_account, duplicate_account) |
|
[Status, Mention, StatusPin, StreamEntry].each do |klass| |
|
klass.where(account_id: duplicate_account.id).in_batches.update_all(account_id: main_account.id) |
|
end |
|
|
|
# Since it's the same remote resource, the remote resource likely |
|
# already believes we are following/blocking, so it's safe to |
|
# re-attribute the relationships too. However, during the presence |
|
# of the index bug users could have *also* followed the reference |
|
# account already, therefore mass update will not work and we need |
|
# to check for (and skip past) uniqueness errors |
|
[Favourite, Follow, FollowRequest, Block, Mute].each do |klass| |
|
klass.where(account_id: duplicate_account.id).find_each do |record| |
|
record.update_attribute(:account_id, main_account.id) |
|
rescue ActiveRecord::RecordNotUnique |
|
next |
|
end |
|
end |
|
|
|
[Follow, FollowRequest, Block, Mute].each do |klass| |
|
klass.where(target_account_id: duplicate_account.id).find_each do |record| |
|
record.update_attribute(:target_account_id, main_account.id) |
|
rescue ActiveRecord::RecordNotUnique |
|
next |
|
end |
|
end |
|
end |
|
end
|
|
|