2019-07-31 22:56:46 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2018-11-18 11:00:15 +05:30
|
|
|
require 'spec_helper'
|
|
|
|
require Rails.root.join('config', 'object_store_settings.rb')
|
|
|
|
|
2023-03-04 22:38:38 +05:30
|
|
|
RSpec.describe ObjectStoreSettings, feature_category: :not_owned do
|
2020-07-28 23:09:34 +05:30
|
|
|
describe '#parse!' do
|
|
|
|
let(:settings) { Settingslogic.new(config) }
|
|
|
|
|
|
|
|
subject { described_class.new(settings).parse! }
|
|
|
|
|
|
|
|
context 'with valid config' do
|
|
|
|
let(:connection) do
|
|
|
|
{
|
|
|
|
'provider' => 'AWS',
|
|
|
|
'aws_access_key_id' => 'AWS_ACCESS_KEY_ID',
|
|
|
|
'aws_secret_access_key' => 'AWS_SECRET_ACCESS_KEY',
|
|
|
|
'region' => 'us-east-1'
|
|
|
|
}
|
|
|
|
end
|
2020-10-24 23:57:45 +05:30
|
|
|
|
2020-07-28 23:09:34 +05:30
|
|
|
let(:config) do
|
|
|
|
{
|
|
|
|
'lfs' => { 'enabled' => true },
|
|
|
|
'artifacts' => { 'enabled' => true },
|
|
|
|
'external_diffs' => { 'enabled' => false },
|
2021-01-03 14:25:43 +05:30
|
|
|
'pages' => { 'enabled' => true },
|
2020-07-28 23:09:34 +05:30
|
|
|
'object_store' => {
|
|
|
|
'enabled' => true,
|
|
|
|
'connection' => connection,
|
|
|
|
'proxy_download' => true,
|
|
|
|
'objects' => {
|
|
|
|
'artifacts' => {
|
|
|
|
'bucket' => 'artifacts',
|
|
|
|
'proxy_download' => false
|
|
|
|
},
|
|
|
|
'lfs' => {
|
|
|
|
'bucket' => 'lfs-objects'
|
|
|
|
},
|
|
|
|
'external_diffs' => {
|
|
|
|
'bucket' => 'external_diffs',
|
|
|
|
'enabled' => false
|
2021-01-03 14:25:43 +05:30
|
|
|
},
|
|
|
|
'pages' => {
|
|
|
|
'bucket' => 'pages'
|
2020-07-28 23:09:34 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
2021-03-11 19:13:27 +05:30
|
|
|
shared_examples 'consolidated settings for objects accelerated by Workhorse' do
|
|
|
|
it 'consolidates active object storage settings' do
|
|
|
|
described_class::WORKHORSE_ACCELERATED_TYPES.each do |object_type|
|
|
|
|
# Use to_h to avoid https://gitlab.com/gitlab-org/gitlab/-/issues/286873
|
|
|
|
section = subject.try(object_type).to_h
|
|
|
|
|
|
|
|
next unless section.dig('object_store', 'enabled')
|
|
|
|
|
|
|
|
expect(section['object_store']['connection']).to eq(connection)
|
|
|
|
expect(section['object_store']['consolidated_settings']).to be true
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-07-28 23:09:34 +05:30
|
|
|
it 'sets correct default values' do
|
|
|
|
subject
|
|
|
|
|
|
|
|
expect(settings.artifacts['enabled']).to be true
|
|
|
|
expect(settings.artifacts['object_store']['enabled']).to be true
|
|
|
|
expect(settings.artifacts['object_store']['connection']).to eq(connection)
|
|
|
|
expect(settings.artifacts['object_store']['direct_upload']).to be true
|
|
|
|
expect(settings.artifacts['object_store']['proxy_download']).to be false
|
|
|
|
expect(settings.artifacts['object_store']['remote_directory']).to eq('artifacts')
|
2022-07-23 23:45:48 +05:30
|
|
|
expect(settings.artifacts['object_store']['bucket_prefix']).to eq(nil)
|
2021-02-22 17:27:13 +05:30
|
|
|
expect(settings.artifacts['object_store']['consolidated_settings']).to be true
|
2021-06-08 01:23:25 +05:30
|
|
|
expect(settings.artifacts).to eq(settings['artifacts'])
|
2020-07-28 23:09:34 +05:30
|
|
|
|
|
|
|
expect(settings.lfs['enabled']).to be true
|
|
|
|
expect(settings.lfs['object_store']['enabled']).to be true
|
|
|
|
expect(settings.lfs['object_store']['connection']).to eq(connection)
|
|
|
|
expect(settings.lfs['object_store']['direct_upload']).to be true
|
|
|
|
expect(settings.lfs['object_store']['proxy_download']).to be true
|
|
|
|
expect(settings.lfs['object_store']['remote_directory']).to eq('lfs-objects')
|
2022-07-23 23:45:48 +05:30
|
|
|
expect(settings.lfs['object_store']['bucket_prefix']).to eq(nil)
|
2021-02-22 17:27:13 +05:30
|
|
|
expect(settings.lfs['object_store']['consolidated_settings']).to be true
|
2021-06-08 01:23:25 +05:30
|
|
|
expect(settings.lfs).to eq(settings['lfs'])
|
2020-07-28 23:09:34 +05:30
|
|
|
|
2021-01-03 14:25:43 +05:30
|
|
|
expect(settings.pages['enabled']).to be true
|
|
|
|
expect(settings.pages['object_store']['enabled']).to be true
|
|
|
|
expect(settings.pages['object_store']['connection']).to eq(connection)
|
|
|
|
expect(settings.pages['object_store']['remote_directory']).to eq('pages')
|
2022-07-23 23:45:48 +05:30
|
|
|
expect(settings.pages['object_store']['bucket_prefix']).to eq(nil)
|
2021-02-22 17:27:13 +05:30
|
|
|
expect(settings.pages['object_store']['consolidated_settings']).to be true
|
2021-06-08 01:23:25 +05:30
|
|
|
expect(settings.pages).to eq(settings['pages'])
|
2021-01-03 14:25:43 +05:30
|
|
|
|
2020-07-28 23:09:34 +05:30
|
|
|
expect(settings.external_diffs['enabled']).to be false
|
2021-03-11 19:13:27 +05:30
|
|
|
expect(settings.external_diffs['object_store']).to be_nil
|
2021-06-08 01:23:25 +05:30
|
|
|
expect(settings.external_diffs).to eq(settings['external_diffs'])
|
2020-07-28 23:09:34 +05:30
|
|
|
end
|
|
|
|
|
2022-07-23 23:45:48 +05:30
|
|
|
it 'supports bucket prefixes' do
|
|
|
|
config['object_store']['objects']['artifacts']['bucket'] = 'gitlab/artifacts'
|
|
|
|
config['object_store']['objects']['lfs']['bucket'] = 'gitlab/lfs'
|
|
|
|
|
|
|
|
subject
|
|
|
|
|
|
|
|
expect(settings.artifacts['object_store']['remote_directory']).to eq('gitlab')
|
|
|
|
expect(settings.artifacts['object_store']['bucket_prefix']).to eq('artifacts')
|
|
|
|
expect(settings.lfs['object_store']['remote_directory']).to eq('gitlab')
|
|
|
|
expect(settings.lfs['object_store']['bucket_prefix']).to eq('lfs')
|
|
|
|
end
|
|
|
|
|
2022-11-25 23:54:43 +05:30
|
|
|
context 'with Google CDN enabled' do
|
|
|
|
let(:cdn_config) do
|
|
|
|
{
|
|
|
|
'provider' => 'Google',
|
|
|
|
'url' => 'https://cdn.example.org',
|
|
|
|
'key_name' => 'stanhu-key',
|
|
|
|
'key' => Base64.urlsafe_encode64(SecureRandom.hex)
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
before do
|
|
|
|
config['object_store']['objects']['artifacts']['cdn'] = cdn_config
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'populates artifacts CDN config' do
|
|
|
|
subject
|
|
|
|
|
|
|
|
expect(settings.artifacts['object_store']['cdn']).to eq(cdn_config)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-07-28 23:09:34 +05:30
|
|
|
it 'raises an error when a bucket is missing' do
|
|
|
|
config['object_store']['objects']['lfs'].delete('bucket')
|
|
|
|
|
|
|
|
expect { subject }.to raise_error(/Object storage for lfs must have a bucket specified/)
|
|
|
|
end
|
|
|
|
|
2021-01-03 14:25:43 +05:30
|
|
|
it 'does not raise error if pages bucket is missing' do
|
|
|
|
config['object_store']['objects']['pages'].delete('bucket')
|
|
|
|
|
|
|
|
expect { subject }.not_to raise_error
|
2021-01-29 00:20:46 +05:30
|
|
|
expect(settings.pages['object_store']).to eq(nil)
|
2021-01-03 14:25:43 +05:30
|
|
|
end
|
|
|
|
|
2021-03-11 19:13:27 +05:30
|
|
|
context 'GitLab Pages' do
|
|
|
|
let(:pages_connection) { { 'provider' => 'Google', 'google_application_default' => true } }
|
|
|
|
|
|
|
|
before do
|
|
|
|
config['pages'] = {
|
2021-02-22 17:27:13 +05:30
|
|
|
'enabled' => true,
|
2021-03-11 19:13:27 +05:30
|
|
|
'object_store' => {
|
|
|
|
'enabled' => true,
|
|
|
|
'connection' => pages_connection
|
|
|
|
}
|
2021-02-22 17:27:13 +05:30
|
|
|
}
|
2021-03-11 19:13:27 +05:30
|
|
|
end
|
2021-02-22 17:27:13 +05:30
|
|
|
|
2021-03-11 19:13:27 +05:30
|
|
|
it_behaves_like 'consolidated settings for objects accelerated by Workhorse'
|
2021-02-22 17:27:13 +05:30
|
|
|
|
2021-03-11 19:13:27 +05:30
|
|
|
it 'allows pages to define its own connection' do
|
|
|
|
expect { subject }.not_to raise_error
|
2021-02-22 17:27:13 +05:30
|
|
|
|
2021-03-11 19:13:27 +05:30
|
|
|
expect(settings.pages['object_store']['connection']).to eq(pages_connection)
|
|
|
|
expect(settings.pages['object_store']['consolidated_settings']).to be_falsey
|
|
|
|
end
|
|
|
|
end
|
2021-02-22 17:27:13 +05:30
|
|
|
|
2021-03-11 19:13:27 +05:30
|
|
|
context 'when object storage is disabled for artifacts with no bucket' do
|
|
|
|
before do
|
|
|
|
config['artifacts'] = {
|
|
|
|
'enabled' => true,
|
|
|
|
'object_store' => {}
|
|
|
|
}
|
|
|
|
config['object_store']['objects']['artifacts'] = {
|
|
|
|
'enabled' => false
|
|
|
|
}
|
2021-02-22 17:27:13 +05:30
|
|
|
end
|
|
|
|
|
2021-03-11 19:13:27 +05:30
|
|
|
it_behaves_like 'consolidated settings for objects accelerated by Workhorse'
|
|
|
|
|
|
|
|
it 'does not enable consolidated settings for artifacts' do
|
|
|
|
subject
|
|
|
|
|
|
|
|
expect(settings.artifacts['enabled']).to be true
|
|
|
|
expect(settings.artifacts['object_store']['remote_directory']).to be_nil
|
2022-07-23 23:45:48 +05:30
|
|
|
expect(settings.artifacts['object_store']['bucket_prefix']).to be_nil
|
2021-03-11 19:13:27 +05:30
|
|
|
expect(settings.artifacts['object_store']['enabled']).to be_falsey
|
|
|
|
expect(settings.artifacts['object_store']['consolidated_settings']).to be_falsey
|
|
|
|
end
|
2021-02-22 17:27:13 +05:30
|
|
|
end
|
|
|
|
|
2020-07-28 23:09:34 +05:30
|
|
|
context 'with legacy config' do
|
|
|
|
let(:legacy_settings) do
|
|
|
|
{
|
|
|
|
'enabled' => true,
|
|
|
|
'remote_directory' => 'some-bucket',
|
2022-07-16 23:28:13 +05:30
|
|
|
'direct_upload' => false,
|
2020-07-28 23:09:34 +05:30
|
|
|
'proxy_download' => false
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
before do
|
2022-07-16 23:28:13 +05:30
|
|
|
settings.lfs['object_store'] = described_class.legacy_parse(legacy_settings, 'lfs')
|
2020-07-28 23:09:34 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
it 'does not alter config if legacy settings are specified' do
|
|
|
|
subject
|
|
|
|
|
|
|
|
expect(settings.artifacts['object_store']).to be_nil
|
|
|
|
expect(settings.lfs['object_store']['remote_directory']).to eq('some-bucket')
|
2022-07-23 23:45:48 +05:30
|
|
|
expect(settings.lfs['object_store']['bucket_prefix']).to eq(nil)
|
2022-07-16 23:28:13 +05:30
|
|
|
expect(settings.lfs['object_store']['direct_upload']).to eq(true)
|
|
|
|
expect(settings.external_diffs['object_store']).to be_nil
|
|
|
|
end
|
|
|
|
end
|
2020-07-28 23:09:34 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '.legacy_parse' do
|
2019-07-07 11:18:12 +05:30
|
|
|
it 'sets correct default values' do
|
2022-07-16 23:28:13 +05:30
|
|
|
settings = described_class.legacy_parse(nil, 'artifacts')
|
2018-11-18 11:00:15 +05:30
|
|
|
|
|
|
|
expect(settings['enabled']).to be false
|
2022-07-16 23:28:13 +05:30
|
|
|
expect(settings['direct_upload']).to be true
|
2018-11-18 11:00:15 +05:30
|
|
|
expect(settings['remote_directory']).to be nil
|
2022-07-23 23:45:48 +05:30
|
|
|
expect(settings['bucket_prefix']).to be nil
|
2018-11-18 11:00:15 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
it 'respects original values' do
|
|
|
|
original_settings = Settingslogic.new({
|
|
|
|
'enabled' => true,
|
|
|
|
'remote_directory' => 'artifacts'
|
|
|
|
})
|
|
|
|
|
2022-07-16 23:28:13 +05:30
|
|
|
settings = described_class.legacy_parse(original_settings, 'artifacts')
|
2018-11-18 11:00:15 +05:30
|
|
|
|
|
|
|
expect(settings['enabled']).to be true
|
2022-07-16 23:28:13 +05:30
|
|
|
expect(settings['direct_upload']).to be true
|
2018-11-18 11:00:15 +05:30
|
|
|
expect(settings['remote_directory']).to eq 'artifacts'
|
2022-07-23 23:45:48 +05:30
|
|
|
expect(settings['bucket_prefix']).to be nil
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'supports bucket prefixes' do
|
|
|
|
original_settings = Settingslogic.new({
|
|
|
|
'enabled' => true,
|
|
|
|
'remote_directory' => 'gitlab/artifacts'
|
|
|
|
})
|
|
|
|
|
|
|
|
settings = described_class.legacy_parse(original_settings, 'artifacts')
|
|
|
|
expect(settings['remote_directory']).to eq 'gitlab'
|
|
|
|
expect(settings['bucket_prefix']).to eq 'artifacts'
|
2018-11-18 11:00:15 +05:30
|
|
|
end
|
2022-07-23 23:45:48 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
describe '.split_bucket_prefix' do
|
|
|
|
using RSpec::Parameterized::TableSyntax
|
|
|
|
|
|
|
|
subject { described_class.split_bucket_prefix(input) }
|
|
|
|
|
|
|
|
context 'valid inputs' do
|
|
|
|
where(:input, :bucket, :prefix) do
|
|
|
|
nil | nil | nil
|
|
|
|
'' | nil | nil
|
|
|
|
'bucket' | 'bucket' | nil
|
|
|
|
'bucket/prefix' | 'bucket' | 'prefix'
|
|
|
|
'bucket/pre/fix' | 'bucket' | 'pre/fix'
|
|
|
|
end
|
|
|
|
|
|
|
|
with_them do
|
|
|
|
it { expect(subject).to eq([bucket, prefix]) }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'invalid inputs' do
|
|
|
|
where(:input) do
|
|
|
|
[
|
|
|
|
['bucket/'],
|
|
|
|
['bucket/.'],
|
|
|
|
['bucket/..'],
|
|
|
|
['bucket/prefix/'],
|
|
|
|
['bucket/prefix/.'],
|
|
|
|
['bucket/prefix/..'],
|
|
|
|
['/bucket/prefix'],
|
|
|
|
['./bucket/prefix'],
|
|
|
|
['../bucket/prefix'],
|
|
|
|
['bucket//prefix'],
|
|
|
|
['bucket/./prefix'],
|
|
|
|
['bucket/../prefix']
|
|
|
|
]
|
|
|
|
end
|
|
|
|
|
|
|
|
with_them do
|
|
|
|
it { expect { subject }.to raise_error(/invalid bucket/) }
|
2022-07-16 23:28:13 +05:30
|
|
|
end
|
|
|
|
end
|
2018-11-18 11:00:15 +05:30
|
|
|
end
|
|
|
|
end
|