Commit 34646406 authored by Sean McGivern's avatar Sean McGivern

Merge branch...

Merge branch '50359-activerecord-statementinvalid-pg-querycanceled-error-canceling-statement-due-to-statement-timeout' into 'master'

Resolve "ActiveRecord::StatementInvalid: PG::QueryCanceled: ERROR:  canceling statement due to statement timeout"

Closes #50359

See merge request gitlab-org/gitlab-ce!21893
parents 58a256ad f5abc2e8
---
title: Fix timeout when running the RemoveRestrictedTodos background migration
merge_request: 21893
author:
type: fixed
# frozen_string_literal: true
# See http://doc.gitlab.com/ce/development/migration_style_guide.html
# for more information on how to write migrations for GitLab.
# rescheduling of the revised RemoveRestrictedTodos background migration
class RemoveRestrictedTodosWithCte < ActiveRecord::Migration
DOWNTIME = false
disable_ddl_transaction!
MIGRATION = 'RemoveRestrictedTodos'.freeze
BATCH_SIZE = 1000
DELAY_INTERVAL = 5.minutes.to_i
class Project < ActiveRecord::Base
include EachBatch
self.table_name = 'projects'
end
def up
Project.where('EXISTS (SELECT 1 FROM todos WHERE todos.project_id = projects.id)')
.each_batch(of: BATCH_SIZE) do |batch, index|
range = batch.pluck('MIN(id)', 'MAX(id)').first
BackgroundMigrationWorker.perform_in(index * DELAY_INTERVAL, MIGRATION, range)
end
end
def down
# nothing to do
end
end
...@@ -11,7 +11,7 @@ ...@@ -11,7 +11,7 @@
# #
# It's strongly recommended that you check this file into your version control system. # It's strongly recommended that you check this file into your version control system.
ActiveRecord::Schema.define(version: 20180924201039) do ActiveRecord::Schema.define(version: 20181002172433) do
# These are extensions that must be enabled in order to support this database # These are extensions that must be enabled in order to support this database
enable_extension "plpgsql" enable_extension "plpgsql"
......
# frozen_string_literal: true # frozen_string_literal: true
# rubocop:disable Style/Documentation # rubocop:disable Style/Documentation
# rubocop:disable Metrics/ClassLength
module Gitlab module Gitlab
module BackgroundMigration module BackgroundMigration
...@@ -49,12 +50,15 @@ module Gitlab ...@@ -49,12 +50,15 @@ module Gitlab
private private
def remove_non_members_todos(project_id) def remove_non_members_todos(project_id)
Todo.where(project_id: project_id) if Gitlab::Database.postgresql?
.where('user_id NOT IN (?)', authorized_users(project_id)) batch_remove_todos_cte(project_id)
else
unauthorized_project_todos(project_id)
.each_batch(of: 5000) do |batch| .each_batch(of: 5000) do |batch|
batch.delete_all batch.delete_all
end end
end end
end
def remove_confidential_issue_todos(project_id) def remove_confidential_issue_todos(project_id)
# min access level to access a confidential issue is reporter # min access level to access a confidential issue is reporter
...@@ -86,12 +90,15 @@ module Gitlab ...@@ -86,12 +90,15 @@ module Gitlab
next if target_types.empty? next if target_types.empty?
Todo.where(project_id: project_id) if Gitlab::Database.postgresql?
.where('user_id NOT IN (?)', authorized_users(project_id)) batch_remove_todos_cte(project_id, target_types)
else
unauthorized_project_todos(project_id)
.where(target_type: target_types) .where(target_type: target_types)
.delete_all .delete_all
end end
end end
end
def private?(feature_level) def private?(feature_level)
feature_level == PRIVATE_FEATURE feature_level == PRIVATE_FEATURE
...@@ -100,6 +107,65 @@ module Gitlab ...@@ -100,6 +107,65 @@ module Gitlab
def authorized_users(project_id) def authorized_users(project_id)
ProjectAuthorization.select(:user_id).where(project_id: project_id) ProjectAuthorization.select(:user_id).where(project_id: project_id)
end end
def unauthorized_project_todos(project_id)
Todo.where(project_id: project_id)
.where('user_id NOT IN (?)', authorized_users(project_id))
end
def batch_remove_todos_cte(project_id, target_types = nil)
loop do
count = remove_todos_cte(project_id, target_types)
break if count == 0
end
end
def remove_todos_cte(project_id, target_types = nil)
sql = []
sql << with_all_todos_sql(project_id, target_types)
sql << as_deleted_sql
sql << "SELECT count(*) FROM deleted"
result = Todo.connection.exec_query(sql.join(' '))
result.rows[0][0].to_i
end
def with_all_todos_sql(project_id, target_types = nil)
if target_types
table = Arel::Table.new(:todos)
in_target = table[:target_type].in(target_types)
target_types_sql = " AND #{in_target.to_sql}"
end
<<-SQL
WITH all_todos AS (
SELECT id
FROM "todos"
WHERE "todos"."project_id" = #{project_id}
AND (user_id NOT IN (
SELECT "project_authorizations"."user_id"
FROM "project_authorizations"
WHERE "project_authorizations"."project_id" = #{project_id})
#{target_types_sql}
)
),
SQL
end
def as_deleted_sql
<<-SQL
deleted AS (
DELETE FROM todos
WHERE id IN (
SELECT id
FROM all_todos
LIMIT 5000
)
RETURNING id
)
SQL
end
end end
end end
end end
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment