Commit fce36270 authored by John Cai's avatar John Cai

Pick repository storage based on weight

Modify the logic for picking a new repository storage to use the
weighted values in repository_storages_weighted instead of the array in
repository_storages.
parent ef878663
...@@ -297,10 +297,21 @@ module ApplicationSettingImplementation ...@@ -297,10 +297,21 @@ module ApplicationSettingImplementation
performance_bar_allowed_group_id.present? performance_bar_allowed_group_id.present?
end end
# Choose one of the available repository storage options. Currently all have def normalized_repository_storage_weights
# equal weighting. strong_memoize(:normalized_repository_storage_weights) do
weights_total = repository_storages_weighted.values.reduce(:+)
repository_storages_weighted.transform_values do |w|
next w if weights_total == 0
w.to_f / weights_total
end
end
end
# Choose one of the available repository storage options based on a normalized weighted probability.
def pick_repository_storage def pick_repository_storage
repository_storages.sample normalized_repository_storage_weights.max_by { |_, weight| rand**(1.0 / weight) }.first
end end
def runners_registration_token def runners_registration_token
......
---
title: Pick repository storage based on weight
merge_request: 34095
author:
type: changed
...@@ -288,11 +288,37 @@ RSpec.shared_examples 'application settings examples' do ...@@ -288,11 +288,37 @@ RSpec.shared_examples 'application settings examples' do
end end
describe '#pick_repository_storage' do describe '#pick_repository_storage' do
it 'uses Array#sample to pick a random storage' do before do
array = double('array', sample: 'random') allow(setting).to receive(:repository_storages_weighted).and_return({ 'default' => 20, 'backup' => 80 })
expect(setting).to receive(:repository_storages).and_return(array) end
it 'chooses repository based on weight' do
picked_storages = { 'default' => 0.0, 'backup' => 0.0 }
10_000.times { picked_storages[setting.pick_repository_storage] += 1 }
expect(((picked_storages['default'] / 10_000) * 100).round.to_i).to be_between(19, 21)
expect(((picked_storages['backup'] / 10_000) * 100).round.to_i).to be_between(79, 81)
end
end
describe '#normalized_repository_storage_weights' do
using RSpec::Parameterized::TableSyntax
expect(setting.pick_repository_storage).to eq('random') where(:storages, :normalized) do
{ 'default' => 0, 'backup' => 100 } | { 'default' => 0.0, 'backup' => 1.0 }
{ 'default' => 100, 'backup' => 100 } | { 'default' => 0.5, 'backup' => 0.5 }
{ 'default' => 20, 'backup' => 80 } | { 'default' => 0.2, 'backup' => 0.8 }
{ 'default' => 0, 'backup' => 0 } | { 'default' => 0.0, 'backup' => 0.0 }
end
with_them do
before do
allow(setting).to receive(:repository_storages_weighted).and_return(storages)
end
it 'normalizes storage weights' do
expect(setting.normalized_repository_storage_weights).to eq(normalized)
end
end end
end end
......
...@@ -148,18 +148,18 @@ RSpec.shared_examples 'model with repository' do ...@@ -148,18 +148,18 @@ RSpec.shared_examples 'model with repository' do
expect(subject).to eq('picked') expect(subject).to eq('picked')
end end
it 'picks from the latest available storage', :request_store do it 'picks from the available storages based on weight', :request_store do
stub_env('IN_MEMORY_APPLICATION_SETTINGS', 'false') stub_env('IN_MEMORY_APPLICATION_SETTINGS', 'false')
Gitlab::CurrentSettings.expire_current_application_settings Gitlab::CurrentSettings.expire_current_application_settings
Gitlab::CurrentSettings.current_application_settings Gitlab::CurrentSettings.current_application_settings
settings = ApplicationSetting.last settings = ApplicationSetting.last
settings.repository_storages = %w(picked) settings.repository_storages_weighted = { 'picked' => 100, 'default' => 0 }
settings.save! settings.save!
expect(Gitlab::CurrentSettings.repository_storages).to eq(%w(default)) expect(Gitlab::CurrentSettings.repository_storages_weighted).to eq({ 'default' => 100 })
expect(subject).to eq('picked') expect(subject).to eq('picked')
expect(Gitlab::CurrentSettings.repository_storages).to eq(%w(picked)) expect(Gitlab::CurrentSettings.repository_storages_weighted).to eq({ 'default' => 0, 'picked' => 100 })
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