Properly expire cache for **all** MRs of a pipeline
Turn ExpirePipelineCacheService into Worker so it can fetch all the merge requests for which the pipeline runs or did run against.
This commit is contained in:
parent
f07edb5af1
commit
ccb9beeed0
5 changed files with 66 additions and 60 deletions
|
@ -96,8 +96,7 @@ module Ci
|
||||||
|
|
||||||
pipeline.run_after_commit do
|
pipeline.run_after_commit do
|
||||||
PipelineHooksWorker.perform_async(id)
|
PipelineHooksWorker.perform_async(id)
|
||||||
Ci::ExpirePipelineCacheService.new(project, nil)
|
ExpirePipelineCacheWorker.perform_async(pipeline.id)
|
||||||
.execute(pipeline)
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -385,6 +384,13 @@ module Ci
|
||||||
.select { |merge_request| merge_request.head_pipeline.try(:id) == self.id }
|
.select { |merge_request| merge_request.head_pipeline.try(:id) == self.id }
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# All the merge requests for which the current pipeline runs/ran against
|
||||||
|
def all_merge_requests
|
||||||
|
@all_merge_requests ||= project.merge_requests
|
||||||
|
.where(source_branch: ref)
|
||||||
|
.select { |merge_request| merge_request.all_pipelines.includes(self) }
|
||||||
|
end
|
||||||
|
|
||||||
def detailed_status(current_user)
|
def detailed_status(current_user)
|
||||||
Gitlab::Ci::Status::Pipeline::Factory
|
Gitlab::Ci::Status::Pipeline::Factory
|
||||||
.new(self, current_user)
|
.new(self, current_user)
|
||||||
|
|
|
@ -1,51 +0,0 @@
|
||||||
module Ci
|
|
||||||
class ExpirePipelineCacheService < BaseService
|
|
||||||
attr_reader :pipeline
|
|
||||||
|
|
||||||
def execute(pipeline)
|
|
||||||
@pipeline = pipeline
|
|
||||||
store = Gitlab::EtagCaching::Store.new
|
|
||||||
|
|
||||||
store.touch(project_pipelines_path)
|
|
||||||
store.touch(commit_pipelines_path) if pipeline.commit
|
|
||||||
store.touch(new_merge_request_pipelines_path)
|
|
||||||
merge_requests_pipelines_paths.each { |path| store.touch(path) }
|
|
||||||
|
|
||||||
Gitlab::Cache::Ci::ProjectPipelineStatus.update_for_pipeline(@pipeline)
|
|
||||||
end
|
|
||||||
|
|
||||||
private
|
|
||||||
|
|
||||||
def project_pipelines_path
|
|
||||||
Gitlab::Routing.url_helpers.namespace_project_pipelines_path(
|
|
||||||
project.namespace,
|
|
||||||
project,
|
|
||||||
format: :json)
|
|
||||||
end
|
|
||||||
|
|
||||||
def commit_pipelines_path
|
|
||||||
Gitlab::Routing.url_helpers.pipelines_namespace_project_commit_path(
|
|
||||||
project.namespace,
|
|
||||||
project,
|
|
||||||
pipeline.commit.id,
|
|
||||||
format: :json)
|
|
||||||
end
|
|
||||||
|
|
||||||
def new_merge_request_pipelines_path
|
|
||||||
Gitlab::Routing.url_helpers.new_namespace_project_merge_request_path(
|
|
||||||
project.namespace,
|
|
||||||
project,
|
|
||||||
format: :json)
|
|
||||||
end
|
|
||||||
|
|
||||||
def merge_requests_pipelines_paths
|
|
||||||
pipeline.merge_requests.collect do |merge_request|
|
|
||||||
Gitlab::Routing.url_helpers.pipelines_namespace_project_merge_request_path(
|
|
||||||
project.namespace,
|
|
||||||
project,
|
|
||||||
merge_request,
|
|
||||||
format: :json)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
51
app/workers/expire_pipeline_cache_worker.rb
Normal file
51
app/workers/expire_pipeline_cache_worker.rb
Normal file
|
@ -0,0 +1,51 @@
|
||||||
|
class ExpirePipelineCacheWorker
|
||||||
|
include Sidekiq::Worker
|
||||||
|
include PipelineQueue
|
||||||
|
|
||||||
|
def perform(id)
|
||||||
|
pipeline = Ci::Pipeline.find(id)
|
||||||
|
project = pipeline.project
|
||||||
|
store = Gitlab::EtagCaching::Store.new
|
||||||
|
|
||||||
|
store.touch(project_pipelines_path(project))
|
||||||
|
store.touch(commit_pipelines_path(project, pipeline.commit)) if pipeline.commit
|
||||||
|
store.touch(new_merge_request_pipelines_path(project))
|
||||||
|
merge_requests_pipelines_paths(project, pipeline).each { |path| store.touch(path) }
|
||||||
|
|
||||||
|
Gitlab::Cache::Ci::ProjectPipelineStatus.update_for_pipeline(pipeline)
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def project_pipelines_path(project)
|
||||||
|
Gitlab::Routing.url_helpers.namespace_project_pipelines_path(
|
||||||
|
project.namespace,
|
||||||
|
project,
|
||||||
|
format: :json)
|
||||||
|
end
|
||||||
|
|
||||||
|
def commit_pipelines_path(project, commit)
|
||||||
|
Gitlab::Routing.url_helpers.pipelines_namespace_project_commit_path(
|
||||||
|
project.namespace,
|
||||||
|
project,
|
||||||
|
commit.id,
|
||||||
|
format: :json)
|
||||||
|
end
|
||||||
|
|
||||||
|
def new_merge_request_pipelines_path(project)
|
||||||
|
Gitlab::Routing.url_helpers.new_namespace_project_merge_request_path(
|
||||||
|
project.namespace,
|
||||||
|
project,
|
||||||
|
format: :json)
|
||||||
|
end
|
||||||
|
|
||||||
|
def merge_requests_pipelines_paths(project, pipeline)
|
||||||
|
pipeline.all_merge_requests.collect do |merge_request|
|
||||||
|
Gitlab::Routing.url_helpers.pipelines_namespace_project_merge_request_path(
|
||||||
|
project.namespace,
|
||||||
|
project,
|
||||||
|
merge_request,
|
||||||
|
format: :json)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
|
@ -376,8 +376,8 @@ describe Ci::Pipeline, models: true do
|
||||||
end
|
end
|
||||||
|
|
||||||
describe 'pipeline caching' do
|
describe 'pipeline caching' do
|
||||||
it 'executes ExpirePipelinesCacheService' do
|
it 'performs ExpirePipelinesCacheWorker' do
|
||||||
expect_any_instance_of(Ci::ExpirePipelineCacheService).to receive(:execute).with(pipeline)
|
expect(ExpirePipelineCacheWorker).to receive(:perform_async).with(pipeline.id)
|
||||||
|
|
||||||
pipeline.cancel
|
pipeline.cancel
|
||||||
end
|
end
|
||||||
|
|
|
@ -1,12 +1,12 @@
|
||||||
require 'spec_helper'
|
require 'spec_helper'
|
||||||
|
|
||||||
describe Ci::ExpirePipelineCacheService, services: true do
|
describe ExpirePipelineCacheWorker do
|
||||||
let(:user) { create(:user) }
|
let(:user) { create(:user) }
|
||||||
let(:project) { create(:empty_project) }
|
let(:project) { create(:empty_project) }
|
||||||
let(:pipeline) { create(:ci_pipeline, project: project) }
|
let(:pipeline) { create(:ci_pipeline, project: project) }
|
||||||
subject { described_class.new(project, user) }
|
subject { described_class.new }
|
||||||
|
|
||||||
describe '#execute' do
|
describe '#perform' do
|
||||||
it 'invalidate Etag caching for project pipelines path' do
|
it 'invalidate Etag caching for project pipelines path' do
|
||||||
pipelines_path = "/#{project.full_path}/pipelines.json"
|
pipelines_path = "/#{project.full_path}/pipelines.json"
|
||||||
new_mr_pipelines_path = "/#{project.full_path}/merge_requests/new.json"
|
new_mr_pipelines_path = "/#{project.full_path}/merge_requests/new.json"
|
||||||
|
@ -14,14 +14,14 @@ describe Ci::ExpirePipelineCacheService, services: true do
|
||||||
expect_any_instance_of(Gitlab::EtagCaching::Store).to receive(:touch).with(pipelines_path)
|
expect_any_instance_of(Gitlab::EtagCaching::Store).to receive(:touch).with(pipelines_path)
|
||||||
expect_any_instance_of(Gitlab::EtagCaching::Store).to receive(:touch).with(new_mr_pipelines_path)
|
expect_any_instance_of(Gitlab::EtagCaching::Store).to receive(:touch).with(new_mr_pipelines_path)
|
||||||
|
|
||||||
subject.execute(pipeline)
|
subject.perform(pipeline.id)
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'updates the cached status for a project' do
|
it 'updates the cached status for a project' do
|
||||||
expect(Gitlab::Cache::Ci::ProjectPipelineStatus).to receive(:update_for_pipeline).
|
expect(Gitlab::Cache::Ci::ProjectPipelineStatus).to receive(:update_for_pipeline).
|
||||||
with(pipeline)
|
with(pipeline)
|
||||||
|
|
||||||
subject.execute(pipeline)
|
subject.perform(pipeline.id)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
Loading…
Reference in a new issue