234 lines
8.1 KiB
Ruby
234 lines
8.1 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
require 'spec_helper'
|
|
|
|
RSpec.describe Gitlab::Database::Partitioning::DetachedPartitionDropper do
|
|
include Database::TableSchemaHelpers
|
|
|
|
subject(:dropper) { described_class.new }
|
|
|
|
let(:connection) { ActiveRecord::Base.connection }
|
|
|
|
def expect_partition_present(name)
|
|
aggregate_failures do
|
|
expect(table_oid(name)).not_to be_nil
|
|
expect(Postgresql::DetachedPartition.find_by(table_name: name)).not_to be_nil
|
|
end
|
|
end
|
|
|
|
def expect_partition_removed(name)
|
|
aggregate_failures do
|
|
expect(table_oid(name)).to be_nil
|
|
expect(Postgresql::DetachedPartition.find_by(table_name: name)).to be_nil
|
|
end
|
|
end
|
|
|
|
before do
|
|
connection.execute(<<~SQL)
|
|
CREATE TABLE referenced_table (
|
|
id bigserial primary key not null
|
|
)
|
|
SQL
|
|
connection.execute(<<~SQL)
|
|
|
|
CREATE TABLE parent_table (
|
|
id bigserial not null,
|
|
referenced_id bigint not null,
|
|
created_at timestamptz not null,
|
|
primary key (id, created_at),
|
|
constraint fk_referenced foreign key (referenced_id) references referenced_table(id)
|
|
) PARTITION BY RANGE(created_at)
|
|
SQL
|
|
end
|
|
|
|
def create_partition(name:, table: 'parent_table', from:, to:, attached:, drop_after:)
|
|
from = from.beginning_of_month
|
|
to = to.beginning_of_month
|
|
full_name = "#{Gitlab::Database::DYNAMIC_PARTITIONS_SCHEMA}.#{name}"
|
|
connection.execute(<<~SQL)
|
|
CREATE TABLE #{full_name}
|
|
PARTITION OF #{table}
|
|
FOR VALUES FROM ('#{from.strftime('%Y-%m-%d')}') TO ('#{to.strftime('%Y-%m-%d')}')
|
|
SQL
|
|
|
|
unless attached
|
|
connection.execute(<<~SQL)
|
|
ALTER TABLE #{table} DETACH PARTITION #{full_name}
|
|
SQL
|
|
end
|
|
|
|
Postgresql::DetachedPartition.create!(table_name: name,
|
|
drop_after: drop_after)
|
|
end
|
|
|
|
describe '#perform' do
|
|
context 'when the partition should not be dropped yet' do
|
|
it 'does not drop the partition' do
|
|
create_partition(name: 'test_partition',
|
|
from: 2.months.ago, to: 1.month.ago,
|
|
attached: false,
|
|
drop_after: 1.day.from_now)
|
|
|
|
dropper.perform
|
|
|
|
expect_partition_present('test_partition')
|
|
end
|
|
end
|
|
|
|
context 'with a partition to drop' do
|
|
before do
|
|
create_partition(name: 'test_partition',
|
|
from: 2.months.ago,
|
|
to: 1.month.ago.beginning_of_month,
|
|
attached: false,
|
|
drop_after: 1.second.ago)
|
|
end
|
|
|
|
it 'drops the partition' do
|
|
dropper.perform
|
|
|
|
expect(table_oid('test_partition')).to be_nil
|
|
end
|
|
|
|
context 'removing foreign keys' do
|
|
it 'removes foreign keys from the table before dropping it' do
|
|
expect(dropper).to receive(:drop_detached_partition).and_wrap_original do |drop_method, partition_name|
|
|
expect(partition_name).to eq('test_partition')
|
|
expect(foreign_key_exists_by_name(partition_name, 'fk_referenced', schema: Gitlab::Database::DYNAMIC_PARTITIONS_SCHEMA)).to be_falsey
|
|
|
|
drop_method.call(partition_name)
|
|
end
|
|
|
|
expect(foreign_key_exists_by_name('test_partition', 'fk_referenced', schema: Gitlab::Database::DYNAMIC_PARTITIONS_SCHEMA)).to be_truthy
|
|
|
|
dropper.perform
|
|
end
|
|
|
|
it 'does not remove foreign keys from the parent table' do
|
|
expect { dropper.perform }.not_to change { foreign_key_exists_by_name('parent_table', 'fk_referenced') }.from(true)
|
|
end
|
|
|
|
context 'when another process drops the foreign key' do
|
|
it 'skips dropping that foreign key' do
|
|
expect(dropper).to receive(:drop_foreign_key_if_present).and_wrap_original do |drop_meth, *args|
|
|
connection.execute('alter table gitlab_partitions_dynamic.test_partition drop constraint fk_referenced;')
|
|
drop_meth.call(*args)
|
|
end
|
|
|
|
dropper.perform
|
|
|
|
expect_partition_removed('test_partition')
|
|
end
|
|
end
|
|
|
|
context 'when another process drops the partition' do
|
|
it 'skips dropping the foreign key' do
|
|
expect(dropper).to receive(:drop_foreign_key_if_present).and_wrap_original do |drop_meth, *args|
|
|
connection.execute('drop table gitlab_partitions_dynamic.test_partition')
|
|
Postgresql::DetachedPartition.where(table_name: 'test_partition').delete_all
|
|
end
|
|
|
|
expect(Gitlab::AppLogger).not_to receive(:error)
|
|
dropper.perform
|
|
end
|
|
end
|
|
end
|
|
|
|
context 'when another process drops the table while the first waits for a lock' do
|
|
it 'skips the table' do
|
|
# First call to .lock is for removing foreign keys
|
|
expect(Postgresql::DetachedPartition).to receive(:lock).once.ordered.and_call_original
|
|
# Rspec's receive_method_chain does not support .and_wrap_original, so we need to nest here.
|
|
expect(Postgresql::DetachedPartition).to receive(:lock).once.ordered.and_wrap_original do |lock_meth|
|
|
locked = lock_meth.call
|
|
expect(locked).to receive(:find_by).and_wrap_original do |find_meth, *find_args|
|
|
# Another process drops the table then deletes this entry
|
|
Postgresql::DetachedPartition.where(*find_args).delete_all
|
|
find_meth.call(*find_args)
|
|
end
|
|
|
|
locked
|
|
end
|
|
|
|
expect(dropper).not_to receive(:drop_one)
|
|
|
|
dropper.perform
|
|
end
|
|
end
|
|
end
|
|
|
|
context 'when the partition to drop is still attached to its table' do
|
|
before do
|
|
create_partition(name: 'test_partition',
|
|
from: 2.months.ago,
|
|
to: 1.month.ago.beginning_of_month,
|
|
attached: true,
|
|
drop_after: 1.second.ago)
|
|
end
|
|
|
|
it 'does not drop the partition, but does remove the DetachedPartition entry' do
|
|
dropper.perform
|
|
aggregate_failures do
|
|
expect(table_oid('test_partition')).not_to be_nil
|
|
expect(Postgresql::DetachedPartition.find_by(table_name: 'test_partition')).to be_nil
|
|
end
|
|
end
|
|
|
|
context 'when another process removes the entry before this process' do
|
|
it 'does nothing' do
|
|
expect(Postgresql::DetachedPartition).to receive(:lock).and_wrap_original do |lock_meth|
|
|
Postgresql::DetachedPartition.delete_all
|
|
lock_meth.call
|
|
end
|
|
|
|
expect(Gitlab::AppLogger).not_to receive(:error)
|
|
|
|
dropper.perform
|
|
|
|
expect(table_oid('test_partition')).not_to be_nil
|
|
end
|
|
end
|
|
end
|
|
|
|
context 'with multiple partitions to drop' do
|
|
before do
|
|
create_partition(name: 'partition_1',
|
|
from: 3.months.ago,
|
|
to: 2.months.ago,
|
|
attached: false,
|
|
drop_after: 1.second.ago)
|
|
|
|
create_partition(name: 'partition_2',
|
|
from: 2.months.ago,
|
|
to: 1.month.ago,
|
|
attached: false,
|
|
drop_after: 1.second.ago)
|
|
end
|
|
|
|
it 'drops both partitions' do
|
|
dropper.perform
|
|
|
|
expect_partition_removed('partition_1')
|
|
expect_partition_removed('partition_2')
|
|
end
|
|
|
|
context 'when the first drop returns an error' do
|
|
it 'still drops the second partition' do
|
|
expect(dropper).to receive(:drop_detached_partition).ordered.and_raise('injected error')
|
|
expect(dropper).to receive(:drop_detached_partition).ordered.and_call_original
|
|
|
|
dropper.perform
|
|
|
|
# We don't know which partition we tried to drop first, so the tests here have to work with either one
|
|
expect(Postgresql::DetachedPartition.count).to eq(1)
|
|
errored_partition_name = Postgresql::DetachedPartition.first!.table_name
|
|
|
|
dropped_partition_name = (%w[partition_1 partition_2] - [errored_partition_name]).first
|
|
expect_partition_present(errored_partition_name)
|
|
expect_partition_removed(dropped_partition_name)
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|