Commit 0fe4ec87 authored by Stan Hu's avatar Stan Hu

Memoize loading of CI variables

When creating a pipeline with many builds, we load group and project
variables repeatedly, which can cause a high number of Redis calls,
database queries, and RAM usage. We can cache this data within
RequestStore to speed up the building of pipelines.

Relates to https://gitlab.com/gitlab-org/gitlab/issues/198694
parent 5aaedef8
...@@ -403,11 +403,15 @@ class Group < Namespace ...@@ -403,11 +403,15 @@ class Group < Namespace
end end
def ci_variables_for(ref, project) def ci_variables_for(ref, project)
list_of_ids = [self] + ancestors cache_key = "ci_variables_for:group:#{self&.id}:project:#{project&.id}:ref:#{ref}"
variables = Ci::GroupVariable.where(group: list_of_ids)
variables = variables.unprotected unless project.protected_for?(ref) ::Gitlab::SafeRequestStore.fetch(cache_key) do
variables = variables.group_by(&:group_id) list_of_ids = [self] + ancestors
list_of_ids.reverse.flat_map { |group| variables[group.id] }.compact variables = Ci::GroupVariable.where(group: list_of_ids)
variables = variables.unprotected unless project.protected_for?(ref)
variables = variables.group_by(&:group_id)
list_of_ids.reverse.flat_map { |group| variables[group.id] }.compact
end
end end
def group_member(user) def group_member(user)
......
...@@ -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