debian-mirror-gitlab/spec/lib/gitlab/database/partitioning/time_partition_spec.rb
2020-08-09 17:44:08 +05:30

174 lines
5.1 KiB
Ruby

# frozen_string_literal: true
require 'spec_helper'
RSpec.describe Gitlab::Database::Partitioning::TimePartition do
describe '.from_sql' do
subject { described_class.from_sql(table, partition_name, definition) }
let(:table) { 'foo' }
let(:partition_name) { 'foo_bar' }
let(:definition) { 'FOR VALUES FROM (\'2020-04-01 00:00:00\') TO (\'2020-05-01 00:00:00\')' }
it 'uses specified table name' do
expect(subject.table).to eq(table)
end
it 'uses specified partition name' do
expect(subject.partition_name).to eq(partition_name)
end
it 'parses start date' do
expect(subject.from).to eq(Date.parse('2020-04-01'))
end
it 'parses end date' do
expect(subject.to).to eq(Date.parse('2020-05-01'))
end
context 'with MINVALUE as a start date' do
let(:definition) { 'FOR VALUES FROM (MINVALUE) TO (\'2020-05-01\')' }
it 'sets from to nil' do
expect(subject.from).to be_nil
end
end
context 'with MAXVALUE as an end date' do
let(:definition) { 'FOR VALUES FROM (\'2020-04-01\') TO (MAXVALUE)' }
it 'raises a NotImplementedError' do
expect { subject }.to raise_error(NotImplementedError)
end
end
end
describe '#partition_name' do
subject { described_class.new(table, from, to, partition_name: partition_name).partition_name }
let(:table) { 'foo' }
let(:from) { '2020-04-01 00:00:00' }
let(:to) { '2020-05-01 00:00:00' }
let(:partition_name) { nil }
it 'uses table as prefix' do
expect(subject).to start_with(table)
end
it 'uses Year-Month (from) as suffix' do
expect(subject).to end_with("_202004")
end
context 'without from date' do
let(:from) { nil }
it 'uses 000000 as suffix for first partition' do
expect(subject).to end_with("_000000")
end
end
context 'with partition name explicitly given' do
let(:partition_name) { "foo_bar" }
it 'uses given partition name' do
expect(subject).to eq(partition_name)
end
end
end
describe '#to_sql' do
subject { described_class.new(table, from, to).to_sql }
let(:table) { 'foo' }
let(:from) { '2020-04-01 00:00:00' }
let(:to) { '2020-05-01 00:00:00' }
it 'transforms to a CREATE TABLE statement' do
expect(subject).to eq(<<~SQL)
CREATE TABLE IF NOT EXISTS "#{Gitlab::Database::DYNAMIC_PARTITIONS_SCHEMA}"."foo_202004"
PARTITION OF "foo"
FOR VALUES FROM ('2020-04-01') TO ('2020-05-01')
SQL
end
context 'without from date' do
let(:from) { nil }
it 'uses MINVALUE instead' do
expect(subject).to eq(<<~SQL)
CREATE TABLE IF NOT EXISTS "#{Gitlab::Database::DYNAMIC_PARTITIONS_SCHEMA}"."foo_000000"
PARTITION OF "foo"
FOR VALUES FROM (MINVALUE) TO ('2020-05-01')
SQL
end
end
end
describe 'object equality - #eql' do
def expect_inequality(actual, other)
expect(actual.eql?(other)).to be_falsey
expect(actual).not_to eq(other)
end
def expect_equality(actual, other)
expect(actual).to eq(other)
expect(actual.eql?(other)).to be_truthy
expect(actual.hash).to eq(other.hash)
end
def make_new(table: 'foo', from: '2020-04-01 00:00:00', to: '2020-05-01 00:00:00', partition_name: 'foo_202004')
described_class.new(table, from, to, partition_name: partition_name)
end
it 'treats objects identical with identical attributes' do
expect_equality(make_new, make_new)
end
it 'different table leads to in-equality' do
expect_inequality(make_new, make_new(table: 'bar'))
end
it 'different from leads to in-equality' do
expect_inequality(make_new, make_new(from: '2020-05-01 00:00:00'))
end
it 'different to leads to in-equality' do
expect_inequality(make_new, make_new(to: '2020-06-01 00:00:00'))
end
it 'different partition_name leads to in-equality' do
expect_inequality(make_new, make_new(partition_name: 'different'))
end
it 'nil partition_name is ignored if auto-generated matches' do
expect_equality(make_new, make_new(partition_name: nil))
end
end
describe 'Comparable, #<=>' do
let(:table) { 'foo' }
it 'sorts by partition name, i.e. by month - MINVALUE partition first' do
partitions = [
described_class.new(table, '2020-04-01', '2020-05-01'),
described_class.new(table, '2020-02-01', '2020-03-01'),
described_class.new(table, nil, '2020-02-01'),
described_class.new(table, '2020-03-01', '2020-04-01')
]
expect(partitions.sort).to eq([
described_class.new(table, nil, '2020-02-01'),
described_class.new(table, '2020-02-01', '2020-03-01'),
described_class.new(table, '2020-03-01', '2020-04-01'),
described_class.new(table, '2020-04-01', '2020-05-01')
])
end
it 'returns nil for partitions of different tables' do
one = described_class.new('foo', '2020-02-01', '2020-03-01')
two = described_class.new('bar', '2020-02-01', '2020-03-01')
expect(one.<=>(two)).to be_nil
end
end
end