Commit e7d458ef authored by Qingyu Zhao's avatar Qingyu Zhao Committed by Stan Hu

LRU object caching for GroupProjectObjectBuilder

It reduces database queries and object creation for labels/milestones
during project import.
  - reduces import time when project heavily uses labels/milestones
  - reduces database pressure
parent ee4a7e56
...@@ -484,3 +484,6 @@ gem 'countries', '~> 3.0' ...@@ -484,3 +484,6 @@ gem 'countries', '~> 3.0'
gem 'retriable', '~> 3.1.2' gem 'retriable', '~> 3.1.2'
gem 'liquid', '~> 4.0' gem 'liquid', '~> 4.0'
# LRU cache
gem 'lru_redux'
...@@ -598,6 +598,7 @@ GEM ...@@ -598,6 +598,7 @@ GEM
loofah (2.4.0) loofah (2.4.0)
crass (~> 1.0.2) crass (~> 1.0.2)
nokogiri (>= 1.5.9) nokogiri (>= 1.5.9)
lru_redux (1.1.0)
lumberjack (1.0.13) lumberjack (1.0.13)
mail (2.7.1) mail (2.7.1)
mini_mime (>= 0.1.1) mini_mime (>= 0.1.1)
...@@ -1263,6 +1264,7 @@ DEPENDENCIES ...@@ -1263,6 +1264,7 @@ DEPENDENCIES
liquid (~> 4.0) liquid (~> 4.0)
lograge (~> 0.5) lograge (~> 0.5)
loofah (~> 2.2) loofah (~> 2.2)
lru_redux
mail_room (~> 0.10.0) mail_room (~> 0.10.0)
marginalia (~> 1.8.0) marginalia (~> 1.8.0)
memory_profiler (~> 0.9) memory_profiler (~> 0.9)
......
---
title: LRU object caching for GroupProjectObjectBuilder
merge_request: 21823
author:
type: performance
...@@ -12,6 +12,13 @@ module Gitlab ...@@ -12,6 +12,13 @@ module Gitlab
# #
# It also adds some logic around Group Labels/Milestones for edge cases. # It also adds some logic around Group Labels/Milestones for edge cases.
class GroupProjectObjectBuilder class GroupProjectObjectBuilder
# Cache keeps 1000 entries at most, 1000 is chosen based on:
# - one cache entry uses around 0.5K memory, 1000 items uses around 500K.
# (leave some buffer it should be less than 1M). It is afforable cost for project import.
# - for projects in Gitlab.com, it seems 1000 entries for labels/milestones is enough.
# For example, gitlab has ~970 labels and 26 milestones.
LRU_CACHE_SIZE = 1000
def self.build(*args) def self.build(*args)
Project.transaction do Project.transaction do
new(*args).find new(*args).find
...@@ -23,17 +30,34 @@ module Gitlab ...@@ -23,17 +30,34 @@ module Gitlab
@attributes = attributes @attributes = attributes
@group = @attributes['group'] @group = @attributes['group']
@project = @attributes['project'] @project = @attributes['project']
if Gitlab::SafeRequestStore.active?
@lru_cache = cache_from_request_store
@cache_key = [klass, attributes]
end
end end
def find def find
return if epic? && group.nil? return if epic? && group.nil?
find_object || klass.create(project_attributes) find_with_cache do
find_object || klass.create(project_attributes)
end
end end
private private
attr_reader :klass, :attributes, :group, :project attr_reader :klass, :attributes, :group, :project, :lru_cache, :cache_key
def find_with_cache
return yield unless lru_cache && cache_key
lru_cache[cache_key] ||= yield
end
def cache_from_request_store
Gitlab::SafeRequestStore[:lru_cache] ||= LruRedux::Cache.new(LRU_CACHE_SIZE)
end
def find_object def find_object
klass.where(where_clause).first klass.where(where_clause).first
......
...@@ -12,6 +12,59 @@ describe Gitlab::ImportExport::GroupProjectObjectBuilder do ...@@ -12,6 +12,59 @@ describe Gitlab::ImportExport::GroupProjectObjectBuilder do
group: create(:group)) group: create(:group))
end end
let(:lru_cache) { subject.send(:lru_cache) }
let(:cache_key) { subject.send(:cache_key) }
context 'request store is not active' do
subject do
described_class.new(Label,
'title' => 'group label',
'project' => project,
'group' => project.group)
end
it 'ignore cache initialize' do
expect(lru_cache).to be_nil
expect(cache_key).to be_nil
end
end
context 'request store is active', :request_store do
subject do
described_class.new(Label,
'title' => 'group label',
'project' => project,
'group' => project.group)
end
it 'initialize cache in memory' do
expect(lru_cache).not_to be_nil
expect(cache_key).not_to be_nil
end
it 'cache object when first time find the object' do
group_label = create(:group_label, name: 'group label', group: project.group)
expect(subject).to receive(:find_object).and_call_original
expect { subject.find }
.to change { lru_cache[cache_key] }
.from(nil).to(group_label)
expect(subject.find).to eq(group_label)
end
it 'read from cache when object has been cached' do
group_label = create(:group_label, name: 'group label', group: project.group)
subject.find
expect(subject).not_to receive(:find_object)
expect { subject.find }.not_to change { lru_cache[cache_key] }
expect(subject.find).to eq(group_label)
end
end
context 'labels' do context 'labels' do
it 'finds the existing group label' do it 'finds the existing group label' do
group_label = create(:group_label, name: 'group label', group: project.group) group_label = create(:group_label, name: 'group label', group: project.group)
......
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