From e97671b81259a2e8bc02babbe94f5ab7aa19fa93 Mon Sep 17 00:00:00 2001 From: Stan Hu Date: Wed, 27 Dec 2017 07:11:54 -0800 Subject: [PATCH] Simplify implementation and allow for batch updates in MySQL --- ...06151218_issues_moved_to_id_foreign_key.rb | 43 +++++++------------ 1 file changed, 16 insertions(+), 27 deletions(-) diff --git a/db/migrate/20171106151218_issues_moved_to_id_foreign_key.rb b/db/migrate/20171106151218_issues_moved_to_id_foreign_key.rb index 6368e5bd9ce..6395462384b 100644 --- a/db/migrate/20171106151218_issues_moved_to_id_foreign_key.rb +++ b/db/migrate/20171106151218_issues_moved_to_id_foreign_key.rb @@ -15,18 +15,26 @@ class IssuesMovedToIdForeignKey < ActiveRecord::Migration self.table_name = 'issues' def self.with_orphaned_moved_to_issues - # Be careful to use a second table here for comparison otherwise we'll null - # out all rows that don't have id == moved_to_id! - where('NOT EXISTS (SELECT true FROM issues b WHERE issues.moved_to_id = b.id)') - .where('moved_to_id IS NOT NULL') + if Gitlab::Database.postgresql? + # Be careful to use a second table here for comparison otherwise we'll null + # out all rows that don't have id == moved.to_id! + where('NOT EXISTS (SELECT true FROM issues B WHERE issues.moved_to_id = B.id)') + .where('moved_to_id IS NOT NULL') + else + # MySQL doesn't allow modification of the same table in a subquery, + # and using a temporary table isn't automatically guaranteed to work + # due to the MySQL query optimizer. See + # https://dev.mysql.com/doc/refman/5.7/en/update.html for more + # details. + joins('LEFT JOIN issues AS b ON issues.moved_to_id = b.id') + .where('issues.moved_to_id IS NOT NULL AND b.id IS NULL') + end end end def up - if Gitlab::Database.postgresql? - postgresql_remove_orphaned_moved_to_ids - else - mysql_remove_orphaned_moved_to_ids + Issue.with_orphaned_moved_to_issues.each_batch(of: 100) do |batch| + batch.update_all(moved_to_id: nil) end add_concurrent_foreign_key( @@ -45,23 +53,4 @@ class IssuesMovedToIdForeignKey < ActiveRecord::Migration remove_foreign_key_without_error(:issues, column: :moved_to_id) remove_concurrent_index(:issues, :moved_to_id) end - - private - - def postgresql_remove_orphaned_moved_to_ids - Issue.with_orphaned_moved_to_issues.each_batch(of: 100) do |batch| - batch.update_all(moved_to_id: nil) - end - end - - # MySQL doesn't allow modification of the same table in a subquery. See - # https://dev.mysql.com/doc/refman/5.7/en/update.html for more details. - def mysql_remove_orphaned_moved_to_ids - execute <<~SQL - UPDATE issues AS a - LEFT JOIN issues AS b ON a.moved_to_id = b.id - SET a.moved_to_id = NULL - WHERE a.moved_to_id IS NOT NULL AND b.id IS NULL; - SQL - end end