Commit 7a66e12c authored by Mayra Cabrera's avatar Mayra Cabrera

Merge branch 'sh-cache-ci-variables' into 'master'

Memoize loading of CI variables

See merge request gitlab-org/gitlab!26147
parents 4c331c26 0fe4ec87
...@@ -406,12 +406,16 @@ class Group < Namespace ...@@ -406,12 +406,16 @@ class Group < Namespace
end end
def ci_variables_for(ref, project) def ci_variables_for(ref, project)
cache_key = "ci_variables_for:group:#{self&.id}:project:#{project&.id}:ref:#{ref}"
::Gitlab::SafeRequestStore.fetch(cache_key) do
list_of_ids = [self] + ancestors list_of_ids = [self] + ancestors
variables = Ci::GroupVariable.where(group: list_of_ids) variables = Ci::GroupVariable.where(group: list_of_ids)
variables = variables.unprotected unless project.protected_for?(ref) variables = variables.unprotected unless project.protected_for?(ref)
variables = variables.group_by(&:group_id) variables = variables.group_by(&:group_id)
list_of_ids.reverse.flat_map { |group| variables[group.id] }.compact list_of_ids.reverse.flat_map { |group| variables[group.id] }.compact
end end
end
def group_member(user) def group_member(user)
if group_members.loaded? if group_members.loaded?
......
...@@ -1963,6 +1963,14 @@ class Project < ApplicationRecord ...@@ -1963,6 +1963,14 @@ class Project < ApplicationRecord
end end
def ci_variables_for(ref:, environment: nil) def ci_variables_for(ref:, environment: nil)
cache_key = "ci_variables_for:project:#{self&.id}:ref:#{ref}:environment:#{environment}"
::Gitlab::SafeRequestStore.fetch(cache_key) do
uncached_ci_variables_for(ref: ref, environment: environment)
end
end
def uncached_ci_variables_for(ref:, environment: nil)
result = if protected_for?(ref) result = if protected_for?(ref)
variables variables
else else
......
---
title: Memoize loading of CI variables
merge_request: 26147
author:
type: performance
...@@ -911,6 +911,16 @@ describe Group do ...@@ -911,6 +911,16 @@ describe Group do
subject { group.ci_variables_for('ref', project) } subject { group.ci_variables_for('ref', project) }
it 'memoizes the result by ref', :request_store do
expect(project).to receive(:protected_for?).with('ref').once.and_return(true)
expect(project).to receive(:protected_for?).with('other').once.and_return(false)
2.times do
expect(group.ci_variables_for('ref', project)).to contain_exactly(ci_variable, protected_variable)
expect(group.ci_variables_for('other', project)).to contain_exactly(ci_variable)
end
end
shared_examples 'ref is protected' do shared_examples 'ref is protected' do
it 'contains all the variables' do it 'contains all the variables' do
is_expected.to contain_exactly(ci_variable, protected_variable) is_expected.to contain_exactly(ci_variable, protected_variable)
......
...@@ -2930,6 +2930,19 @@ describe Project do ...@@ -2930,6 +2930,19 @@ describe Project do
end end
end end
it 'memoizes the result by ref and environment', :request_store do
scoped_variable = create(:ci_variable, value: 'secret', project: project, environment_scope: 'scoped')
expect(project).to receive(:protected_for?).with('ref').once.and_return(true)
expect(project).to receive(:protected_for?).with('other').twice.and_return(false)
2.times do
expect(project.reload.ci_variables_for(ref: 'ref', environment: 'production')).to contain_exactly(ci_variable, protected_variable)
expect(project.reload.ci_variables_for(ref: 'other')).to contain_exactly(ci_variable)
expect(project.reload.ci_variables_for(ref: 'other', environment: 'scoped')).to contain_exactly(ci_variable, scoped_variable)
end
end
context 'when the ref is not protected' do context 'when the ref is not protected' do
before do before do
allow(project).to receive(:protected_for?).with('ref').and_return(false) allow(project).to receive(:protected_for?).with('ref').and_return(false)
......
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