Use `upsert_all` and `insert_all` to reduce back-and-forth in costly migrations (#29752)
parent
9c24f2d6b1
commit
de740dfb9c
|
@ -4,46 +4,50 @@ class MigrateInteractionSettingsToPolicy < ActiveRecord::Migration[7.1]
|
||||||
disable_ddl_transaction!
|
disable_ddl_transaction!
|
||||||
|
|
||||||
# Dummy classes, to make migration possible across version changes
|
# Dummy classes, to make migration possible across version changes
|
||||||
class Account < ApplicationRecord
|
|
||||||
has_one :user, inverse_of: :account
|
|
||||||
has_one :notification_policy, inverse_of: :account
|
|
||||||
end
|
|
||||||
|
|
||||||
class User < ApplicationRecord
|
class User < ApplicationRecord
|
||||||
belongs_to :account
|
belongs_to :notification_policy, foreign_key: 'account_id', primary_key: 'account_id', optional: true, inverse_of: false
|
||||||
end
|
end
|
||||||
|
|
||||||
class NotificationPolicy < ApplicationRecord
|
class NotificationPolicy < ApplicationRecord; end
|
||||||
belongs_to :account
|
|
||||||
end
|
|
||||||
|
|
||||||
def up
|
def up
|
||||||
User.includes(account: :notification_policy).find_each do |user|
|
User.includes(:notification_policy).in_batches do |users|
|
||||||
deserialized_settings = Oj.load(user.attributes_before_type_cast['settings'])
|
NotificationPolicy.upsert_all(users.filter_map { |user| policy_for_user(user) })
|
||||||
|
|
||||||
next if deserialized_settings.nil?
|
|
||||||
|
|
||||||
policy = user.account.notification_policy || user.account.build_notification_policy
|
|
||||||
requires_new_policy = false
|
|
||||||
|
|
||||||
if deserialized_settings['interactions.must_be_follower']
|
|
||||||
policy.filter_not_followers = true
|
|
||||||
requires_new_policy = true
|
|
||||||
end
|
|
||||||
|
|
||||||
if deserialized_settings['interactions.must_be_following']
|
|
||||||
policy.filter_not_following = true
|
|
||||||
requires_new_policy = true
|
|
||||||
end
|
|
||||||
|
|
||||||
unless deserialized_settings['interactions.must_be_following_dm']
|
|
||||||
policy.filter_private_mentions = false
|
|
||||||
requires_new_policy = true
|
|
||||||
end
|
|
||||||
|
|
||||||
policy.save if requires_new_policy && policy.changed?
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def down; end
|
def down; end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def policy_for_user(user)
|
||||||
|
deserialized_settings = Oj.load(user.attributes_before_type_cast['settings'])
|
||||||
|
return if deserialized_settings.nil?
|
||||||
|
|
||||||
|
requires_new_policy = false
|
||||||
|
|
||||||
|
policy = {
|
||||||
|
account_id: user.account_id,
|
||||||
|
filter_not_followers: false,
|
||||||
|
filter_not_following: false,
|
||||||
|
filter_private_mentions: true,
|
||||||
|
}
|
||||||
|
|
||||||
|
if deserialized_settings['interactions.must_be_follower']
|
||||||
|
policy[:filter_not_followers] = true
|
||||||
|
requires_new_policy = true
|
||||||
|
end
|
||||||
|
|
||||||
|
if deserialized_settings['interactions.must_be_following']
|
||||||
|
policy[:filter_not_following] = true
|
||||||
|
requires_new_policy = true
|
||||||
|
end
|
||||||
|
|
||||||
|
unless deserialized_settings['interactions.must_be_following_dm']
|
||||||
|
policy[:filter_private_mentions] = false
|
||||||
|
requires_new_policy = true
|
||||||
|
end
|
||||||
|
|
||||||
|
policy if requires_new_policy
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -4,51 +4,51 @@ class MigrateInteractionSettingsToPolicyAgain < ActiveRecord::Migration[7.1]
|
||||||
disable_ddl_transaction!
|
disable_ddl_transaction!
|
||||||
|
|
||||||
# Dummy classes, to make migration possible across version changes
|
# Dummy classes, to make migration possible across version changes
|
||||||
class Account < ApplicationRecord
|
|
||||||
has_one :user, inverse_of: :account
|
|
||||||
has_one :notification_policy, inverse_of: :account
|
|
||||||
end
|
|
||||||
|
|
||||||
class User < ApplicationRecord
|
class User < ApplicationRecord
|
||||||
belongs_to :account
|
belongs_to :notification_policy, foreign_key: 'account_id', primary_key: 'account_id', optional: true, inverse_of: false
|
||||||
end
|
end
|
||||||
|
|
||||||
class NotificationPolicy < ApplicationRecord
|
class NotificationPolicy < ApplicationRecord; end
|
||||||
belongs_to :account
|
|
||||||
end
|
|
||||||
|
|
||||||
def up
|
def up
|
||||||
User.includes(account: :notification_policy).find_each do |user|
|
User.includes(:notification_policy).in_batches do |users|
|
||||||
deserialized_settings = Oj.load(user.attributes_before_type_cast['settings'])
|
NotificationPolicy.insert_all(users.filter_map { |user| policy_for_user(user) })
|
||||||
|
|
||||||
next if deserialized_settings.nil?
|
|
||||||
|
|
||||||
# If the user has configured a notification policy, don't override it
|
|
||||||
next if user.account.notification_policy.present?
|
|
||||||
|
|
||||||
policy = user.account.build_notification_policy
|
|
||||||
requires_new_policy = false
|
|
||||||
|
|
||||||
if deserialized_settings['interactions.must_be_follower']
|
|
||||||
policy.filter_not_followers = true
|
|
||||||
requires_new_policy = true
|
|
||||||
end
|
|
||||||
|
|
||||||
if deserialized_settings['interactions.must_be_following']
|
|
||||||
policy.filter_not_following = true
|
|
||||||
requires_new_policy = true
|
|
||||||
end
|
|
||||||
|
|
||||||
unless deserialized_settings['interactions.must_be_following_dm']
|
|
||||||
policy.filter_private_mentions = false
|
|
||||||
requires_new_policy = true
|
|
||||||
end
|
|
||||||
|
|
||||||
policy.save if requires_new_policy && policy.changed?
|
|
||||||
rescue ActiveRecord::RecordNotUnique
|
|
||||||
next
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def down; end
|
def down; end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def policy_for_user(user)
|
||||||
|
deserialized_settings = Oj.load(user.attributes_before_type_cast['settings'])
|
||||||
|
return if deserialized_settings.nil?
|
||||||
|
return if user.notification_policy.present?
|
||||||
|
|
||||||
|
requires_new_policy = false
|
||||||
|
|
||||||
|
policy = {
|
||||||
|
account_id: user.account_id,
|
||||||
|
filter_not_followers: false,
|
||||||
|
filter_not_following: false,
|
||||||
|
filter_private_mentions: true,
|
||||||
|
}
|
||||||
|
|
||||||
|
if deserialized_settings['interactions.must_be_follower']
|
||||||
|
policy[:filter_not_followers] = true
|
||||||
|
requires_new_policy = true
|
||||||
|
end
|
||||||
|
|
||||||
|
if deserialized_settings['interactions.must_be_following']
|
||||||
|
policy[:filter_not_following] = true
|
||||||
|
requires_new_policy = true
|
||||||
|
end
|
||||||
|
|
||||||
|
unless deserialized_settings['interactions.must_be_following_dm']
|
||||||
|
policy[:filter_private_mentions] = false
|
||||||
|
requires_new_policy = true
|
||||||
|
end
|
||||||
|
|
||||||
|
policy if requires_new_policy
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
Loading…
Reference in New Issue