175 lines
5.1 KiB
Ruby
175 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
|