Commit 774a3283 authored by Luke Duncalfe's avatar Luke Duncalfe

Merge branch 'sh-flush-statistics-cache' into 'master'

Flush statistics cache anytime it is updated

See merge request gitlab-org/gitlab!52938
parents c96f3dae bddd15bf
...@@ -2,18 +2,49 @@ ...@@ -2,18 +2,49 @@
module Projects module Projects
class UpdateStatisticsService < BaseService class UpdateStatisticsService < BaseService
include ::Gitlab::Utils::StrongMemoize
STAT_TO_CACHED_METHOD = {
repository_size: :size,
commit_count: :commit_count
}.freeze
def execute def execute
return unless project return unless project
Gitlab::AppLogger.info("Updating statistics for project #{project.id}") Gitlab::AppLogger.info("Updating statistics for project #{project.id}")
project.statistics.refresh!(only: statistics.map(&:to_sym)) expire_repository_caches
expire_wiki_caches
project.statistics.refresh!(only: statistics)
end end
private private
def expire_repository_caches
if statistics.empty?
project.repository.expire_statistics_caches
elsif method_caches_to_expire.present?
project.repository.expire_method_caches(method_caches_to_expire)
end
end
def expire_wiki_caches
return unless project.wiki_enabled? && statistics.include?(:wiki_size)
project.wiki.repository.expire_method_caches([:size])
end
def method_caches_to_expire
strong_memoize(:method_caches_to_expire) do
statistics.map { |stat| STAT_TO_CACHED_METHOD[stat] }.compact
end
end
def statistics def statistics
params[:statistics] strong_memoize(:statistics) do
params[:statistics]&.map(&:to_sym)
end
end end
end end
end end
---
title: Flush statistics cache anytime it is updated
merge_request: 52938
author:
type: fixed
...@@ -3,6 +3,8 @@ ...@@ -3,6 +3,8 @@
require 'spec_helper' require 'spec_helper'
RSpec.describe Projects::UpdateStatisticsService do RSpec.describe Projects::UpdateStatisticsService do
using RSpec::Parameterized::TableSyntax
let(:service) { described_class.new(project, nil, statistics: statistics)} let(:service) { described_class.new(project, nil, statistics: statistics)}
let(:statistics) { %w(repository_size) } let(:statistics) { %w(repository_size) }
...@@ -18,12 +20,46 @@ RSpec.describe Projects::UpdateStatisticsService do ...@@ -18,12 +20,46 @@ RSpec.describe Projects::UpdateStatisticsService do
end end
context 'with an existing project' do context 'with an existing project' do
let(:project) { create(:project) } let_it_be(:project) { create(:project) }
where(:statistics, :method_caches) do
[] | %i(size commit_count)
['repository_size'] | [:size]
[:repository_size] | [:size]
[:lfs_objects_size] | nil
[:commit_count] | [:commit_count] # rubocop:disable Lint/BinaryOperatorWithIdenticalOperands
[:repository_size, :commit_count] | %i(size commit_count)
[:repository_size, :commit_count, :lfs_objects_size] | %i(size commit_count)
end
with_them do
it 'refreshes the project statistics' do
expect(project.statistics).to receive(:refresh!).with(only: statistics.map(&:to_sym)).and_call_original
service.execute
end
it 'invalidates the method caches after a refresh' do
expect(project.wiki.repository).not_to receive(:expire_method_caches)
if method_caches.present?
expect(project.repository).to receive(:expire_method_caches).with(method_caches).and_call_original
else
expect(project.repository).not_to receive(:expire_method_caches)
end
service.execute
end
end
end
context 'with an existing project with a Wiki' do
let(:project) { create(:project, :repository, :wiki_enabled) }
let(:statistics) { [:wiki_size] }
it 'refreshes the project statistics' do it 'invalidates and refreshes Wiki size' do
expect_any_instance_of(ProjectStatistics).to receive(:refresh!) expect(project.statistics).to receive(:refresh!).with(only: statistics).and_call_original
.with(only: statistics.map(&:to_sym)) expect(project.wiki.repository).to receive(:expire_method_caches).with(%i(size)).and_call_original
.and_call_original
service.execute service.execute
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