debian-mirror-gitlab/spec/services/ci/retry_pipeline_service_spec.rb

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

504 lines
17 KiB
Ruby
Raw Permalink Normal View History

2019-07-31 22:56:46 +05:30
# frozen_string_literal: true
2017-08-17 22:00:37 +05:30
require 'spec_helper'
2023-05-27 22:25:52 +05:30
RSpec.describe Ci::RetryPipelineService, '#execute', feature_category: :continuous_integration do
2018-11-08 19:23:39 +05:30
include ProjectForksHelper
2023-01-13 00:05:48 +05:30
let_it_be_with_refind(:user) { create(:user) }
let_it_be_with_refind(:project) { create(:project) }
2017-08-17 22:00:37 +05:30
let(:pipeline) { create(:ci_pipeline, project: project) }
2022-10-11 01:57:18 +05:30
let(:build_stage) { create(:ci_stage, name: 'build', position: 0, pipeline: pipeline) }
let(:test_stage) { create(:ci_stage, name: 'test', position: 1, pipeline: pipeline) }
let(:deploy_stage) { create(:ci_stage, name: 'deploy', position: 2, pipeline: pipeline) }
2017-08-17 22:00:37 +05:30
2023-01-13 00:05:48 +05:30
subject(:service) { described_class.new(project, user) }
2017-08-17 22:00:37 +05:30
context 'when user has full ability to modify pipeline' do
before do
project.add_developer(user)
2023-07-09 08:55:56 +05:30
create(:protected_branch, :developers_can_merge, name: pipeline.ref, project: project)
2017-08-17 22:00:37 +05:30
end
context 'when there are already retried jobs present' do
before do
2022-10-11 01:57:18 +05:30
create_build('rspec', :canceled, build_stage, retried: true)
create_build('rspec', :failed, build_stage)
2017-08-17 22:00:37 +05:30
end
it 'does not retry jobs that has already been retried' do
expect(statuses.first).to be_retried
expect { service.execute(pipeline) }
.to change { CommitStatus.count }.by(1)
end
end
context 'when there are failed builds in the last stage' do
before do
2022-10-11 01:57:18 +05:30
create_build('rspec 1', :success, build_stage)
create_build('rspec 2', :failed, test_stage)
create_build('rspec 3', :canceled, test_stage)
2017-08-17 22:00:37 +05:30
end
it 'enqueues all builds in the last stage' do
service.execute(pipeline)
expect(build('rspec 2')).to be_pending
expect(build('rspec 3')).to be_pending
expect(pipeline.reload).to be_running
end
end
context 'when there are failed or canceled builds in the first stage' do
before do
2022-10-11 01:57:18 +05:30
create_build('rspec 1', :failed, build_stage)
create_build('rspec 2', :canceled, build_stage)
create_build('rspec 3', :canceled, test_stage)
create_build('spinach 1', :canceled, deploy_stage)
2017-08-17 22:00:37 +05:30
end
it 'retries builds failed builds and marks subsequent for processing' do
service.execute(pipeline)
expect(build('rspec 1')).to be_pending
expect(build('rspec 2')).to be_pending
expect(build('rspec 3')).to be_created
expect(build('spinach 1')).to be_created
expect(pipeline.reload).to be_running
end
2021-03-08 18:12:59 +05:30
it 'changes ownership of subsequent builds' do
expect(build('rspec 2').user).not_to eq(user)
expect(build('rspec 3').user).not_to eq(user)
expect(build('spinach 1').user).not_to eq(user)
service.execute(pipeline)
expect(build('rspec 2').user).to eq(user)
expect(build('rspec 3').user).to eq(user)
expect(build('spinach 1').user).to eq(user)
end
2017-08-17 22:00:37 +05:30
end
context 'when there is failed build present which was run on failure' do
before do
2022-10-11 01:57:18 +05:30
create_build('rspec 1', :failed, build_stage)
create_build('rspec 2', :canceled, build_stage)
create_build('rspec 3', :canceled, test_stage)
create_build('report 1', :failed, deploy_stage)
2017-08-17 22:00:37 +05:30
end
it 'retries builds only in the first stage' do
service.execute(pipeline)
expect(build('rspec 1')).to be_pending
expect(build('rspec 2')).to be_pending
expect(build('rspec 3')).to be_created
expect(build('report 1')).to be_created
expect(pipeline.reload).to be_running
end
it 'creates a new job for report job in this case' do
service.execute(pipeline)
2020-04-22 19:07:51 +05:30
expect(statuses.find_by(name: 'report 1', status: 'failed')).to be_retried
2017-08-17 22:00:37 +05:30
end
end
2020-01-01 13:55:28 +05:30
context 'when there is a failed test in a DAG' do
before do
2022-10-11 01:57:18 +05:30
create_build('build', :success, build_stage)
create_build('build2', :success, build_stage)
test_build = create_build('test', :failed, test_stage, scheduling_type: :dag)
2020-01-01 13:55:28 +05:30
create(:ci_build_need, build: test_build, name: 'build')
create(:ci_build_need, build: test_build, name: 'build2')
end
it 'retries the test' do
service.execute(pipeline)
expect(build('build')).to be_success
expect(build('build2')).to be_success
expect(build('test')).to be_pending
expect(build('test').needs.map(&:name)).to match_array(%w(build build2))
end
2020-03-13 15:44:24 +05:30
context 'when there is a failed DAG test without needs' do
before do
2022-10-11 01:57:18 +05:30
create_build('deploy', :failed, deploy_stage, scheduling_type: :dag)
2020-03-13 15:44:24 +05:30
end
it 'retries the test' do
service.execute(pipeline)
expect(build('build')).to be_success
expect(build('build2')).to be_success
expect(build('test')).to be_pending
expect(build('deploy')).to be_pending
end
end
2020-01-01 13:55:28 +05:30
end
2022-05-07 20:08:51 +05:30
context 'when the last stage was skipped' do
2017-08-17 22:00:37 +05:30
before do
2022-10-11 01:57:18 +05:30
create_build('build 1', :success, build_stage)
create_build('test 2', :failed, test_stage)
create_build('report 3', :skipped, deploy_stage)
create_build('report 4', :skipped, deploy_stage)
2017-08-17 22:00:37 +05:30
end
it 'retries builds only in the first stage' do
service.execute(pipeline)
expect(build('build 1')).to be_success
expect(build('test 2')).to be_pending
expect(build('report 3')).to be_created
expect(build('report 4')).to be_created
expect(pipeline.reload).to be_running
end
end
context 'when pipeline contains manual actions' do
context 'when there are optional manual actions only' do
context 'when there is a canceled manual action in first stage' do
before do
2022-10-11 01:57:18 +05:30
create_build('rspec 1', :failed, build_stage)
create_build('staging', :canceled, build_stage, when: :manual, allow_failure: true)
create_build('rspec 2', :canceled, test_stage)
2017-08-17 22:00:37 +05:30
end
it 'retries failed builds and marks subsequent for processing' do
service.execute(pipeline)
expect(build('rspec 1')).to be_pending
expect(build('staging')).to be_manual
expect(build('rspec 2')).to be_created
expect(pipeline.reload).to be_running
end
2021-03-08 18:12:59 +05:30
it 'changes ownership of subsequent builds' do
expect(build('staging').user).not_to eq(user)
expect(build('rspec 2').user).not_to eq(user)
service.execute(pipeline)
expect(build('staging').user).to eq(user)
expect(build('rspec 2').user).to eq(user)
end
2017-08-17 22:00:37 +05:30
end
end
context 'when pipeline has blocking manual actions defined' do
context 'when pipeline retry should enqueue builds' do
before do
2022-10-11 01:57:18 +05:30
create_build('test', :failed, build_stage)
create_build('deploy', :canceled, build_stage, when: :manual, allow_failure: false)
create_build('verify', :canceled, test_stage)
2017-08-17 22:00:37 +05:30
end
it 'retries failed builds' do
service.execute(pipeline)
expect(build('test')).to be_pending
expect(build('deploy')).to be_manual
expect(build('verify')).to be_created
expect(pipeline.reload).to be_running
end
end
context 'when pipeline retry should block pipeline immediately' do
before do
2022-10-11 01:57:18 +05:30
create_build('test', :success, build_stage)
create_build('deploy:1', :success, test_stage, when: :manual, allow_failure: false)
create_build('deploy:2', :failed, test_stage, when: :manual, allow_failure: false)
create_build('verify', :canceled, deploy_stage)
2017-08-17 22:00:37 +05:30
end
it 'reprocesses blocking manual action and blocks pipeline' do
service.execute(pipeline)
expect(build('deploy:1')).to be_success
expect(build('deploy:2')).to be_manual
expect(build('verify')).to be_created
expect(pipeline.reload).to be_blocked
end
end
end
context 'when there is a skipped manual action in last stage' do
before do
2022-10-11 01:57:18 +05:30
create_build('rspec 1', :canceled, build_stage)
create_build('rspec 2', :skipped, build_stage, when: :manual, allow_failure: true)
create_build('staging', :skipped, test_stage, when: :manual, allow_failure: true)
2017-08-17 22:00:37 +05:30
end
it 'retries canceled job and reprocesses manual actions' do
service.execute(pipeline)
expect(build('rspec 1')).to be_pending
expect(build('rspec 2')).to be_manual
expect(build('staging')).to be_created
expect(pipeline.reload).to be_running
end
end
context 'when there is a created manual action in the last stage' do
before do
2022-10-11 01:57:18 +05:30
create_build('rspec 1', :canceled, build_stage)
create_build('staging', :created, test_stage, when: :manual, allow_failure: true)
2017-08-17 22:00:37 +05:30
end
it 'retries canceled job and does not update the manual action' do
service.execute(pipeline)
expect(build('rspec 1')).to be_pending
expect(build('staging')).to be_created
expect(pipeline.reload).to be_running
end
end
context 'when there is a created manual action in the first stage' do
before do
2022-10-11 01:57:18 +05:30
create_build('rspec 1', :canceled, build_stage)
create_build('staging', :created, build_stage, when: :manual, allow_failure: true)
2017-08-17 22:00:37 +05:30
end
it 'retries canceled job and processes the manual action' do
service.execute(pipeline)
expect(build('rspec 1')).to be_pending
expect(build('staging')).to be_manual
expect(pipeline.reload).to be_running
end
end
2023-01-13 00:05:48 +05:30
context 'when there is a failed manual action' do
before do
create_build('rspec', :success, build_stage)
create_build('manual-rspec', :failed, build_stage, when: :manual, allow_failure: true)
end
it 'processes the manual action' do
service.execute(pipeline)
expect(build('rspec')).to be_success
expect(build('manual-rspec')).to be_manual
expect(pipeline.reload).to be_success
end
end
2017-08-17 22:00:37 +05:30
end
it 'closes all todos about failed jobs for pipeline' do
2021-04-29 21:17:54 +05:30
expect(::MergeRequests::AddTodoWhenBuildFailsService)
2017-08-17 22:00:37 +05:30
.to receive_message_chain(:new, :close_all)
service.execute(pipeline)
end
it 'reprocesses the pipeline' do
2020-01-01 13:55:28 +05:30
expect_any_instance_of(Ci::ProcessPipelineService).to receive(:execute)
2017-08-17 22:00:37 +05:30
service.execute(pipeline)
end
2020-05-24 23:13:21 +05:30
context 'when pipeline has processables with nil scheduling_type' do
2022-10-11 01:57:18 +05:30
let!(:build1) { create_build('build1', :success, build_stage) }
let!(:build2) { create_build('build2', :failed, build_stage) }
let!(:build3) { create_build('build3', :failed, test_stage) }
2020-05-24 23:13:21 +05:30
let!(:build3_needs_build1) { create(:ci_build_need, build: build3, name: build1.name) }
before do
statuses.update_all(scheduling_type: nil)
end
it 'populates scheduling_type of processables' do
service.execute(pipeline)
expect(build1.reload.scheduling_type).to eq('stage')
expect(build2.reload.scheduling_type).to eq('stage')
expect(build3.reload.scheduling_type).to eq('dag')
end
end
2020-11-24 15:15:51 +05:30
context 'when the pipeline is a downstream pipeline and the bridge is depended' do
let!(:bridge) { create(:ci_bridge, :strategy_depend, status: 'success') }
before do
create(:ci_sources_pipeline, pipeline: pipeline, source_job: bridge)
end
2022-11-25 23:54:43 +05:30
context 'without permission' do
it 'does nothing to the bridge' do
expect { service.execute(pipeline) }.to not_change { bridge.reload.status }
.and not_change { bridge.reload.user }
end
end
context 'with permission' do
let!(:bridge_pipeline) { create(:ci_pipeline, project: create(:project)) }
let!(:bridge) do
create(:ci_bridge, :strategy_depend, status: 'success', pipeline: bridge_pipeline)
end
2020-11-24 15:15:51 +05:30
2022-11-25 23:54:43 +05:30
before do
bridge_pipeline.project.add_maintainer(user)
end
it 'marks source bridge as pending' do
expect { service.execute(pipeline) }.to change { bridge.reload.status }.to('pending')
2023-03-04 22:38:38 +05:30
end
it 'assigns the current user to the source bridge' do
expect { service.execute(pipeline) }.to change { bridge.reload.user }.to(user)
2022-11-25 23:54:43 +05:30
end
2020-11-24 15:15:51 +05:30
end
end
2021-11-18 22:05:49 +05:30
context 'when there are skipped jobs in later stages' do
before do
2022-10-11 01:57:18 +05:30
create_build('build 1', :success, build_stage)
create_build('test 2', :failed, test_stage)
create_build('report 3', :skipped, deploy_stage)
create_bridge('deploy 4', :skipped, deploy_stage)
2021-11-18 22:05:49 +05:30
end
it 'retries failed jobs and processes skipped jobs' do
service.execute(pipeline)
expect(build('build 1')).to be_success
expect(build('test 2')).to be_pending
expect(build('report 3')).to be_created
expect(build('deploy 4')).to be_created
expect(pipeline.reload).to be_running
end
end
2022-05-07 20:08:51 +05:30
context 'when user is not allowed to retry build' do
before do
build = create(:ci_build, pipeline: pipeline, status: :failed)
2022-06-21 17:19:12 +05:30
allow_next_instance_of(Ci::RetryJobService) do |service|
2022-05-07 20:08:51 +05:30
allow(service).to receive(:can?).with(user, :update_build, build).and_return(false)
end
end
it 'returns an error' do
response = service.execute(pipeline)
expect(response.http_status).to eq(:forbidden)
expect(response.errors).to include('403 Forbidden')
expect(pipeline.reload).not_to be_running
end
end
2017-08-17 22:00:37 +05:30
end
context 'when user is not allowed to retry pipeline' do
2022-05-07 20:08:51 +05:30
it 'returns an error' do
response = service.execute(pipeline)
expect(response.http_status).to eq(:forbidden)
expect(response.errors).to include('403 Forbidden')
expect(pipeline.reload).not_to be_running
2017-08-17 22:00:37 +05:30
end
end
context 'when user is not allowed to trigger manual action' do
before do
project.add_developer(user)
2023-07-09 08:55:56 +05:30
create(:protected_branch, :maintainers_can_push, name: pipeline.ref, project: project)
2017-08-17 22:00:37 +05:30
end
context 'when there is a failed manual action present' do
before do
2022-10-11 01:57:18 +05:30
create_build('test', :failed, build_stage)
create_build('deploy', :failed, build_stage, when: :manual)
create_build('verify', :canceled, test_stage)
2017-08-17 22:00:37 +05:30
end
2022-05-07 20:08:51 +05:30
it 'returns an error' do
response = service.execute(pipeline)
expect(response.http_status).to eq(:forbidden)
expect(response.errors).to include('403 Forbidden')
expect(pipeline.reload).not_to be_running
2017-08-17 22:00:37 +05:30
end
end
context 'when there is a failed manual action in later stage' do
before do
2022-10-11 01:57:18 +05:30
create_build('test', :failed, build_stage)
create_build('deploy', :failed, test_stage, when: :manual)
create_build('verify', :canceled, deploy_stage)
2017-08-17 22:00:37 +05:30
end
2022-05-07 20:08:51 +05:30
it 'returns an error' do
response = service.execute(pipeline)
expect(response.http_status).to eq(:forbidden)
expect(response.errors).to include('403 Forbidden')
expect(pipeline.reload).not_to be_running
2017-08-17 22:00:37 +05:30
end
end
end
2018-11-08 19:23:39 +05:30
context 'when maintainer is allowed to push to forked project' do
let(:user) { create(:user) }
let(:project) { create(:project, :public) }
let(:forked_project) { fork_project(project) }
let(:pipeline) { create(:ci_pipeline, project: forked_project, ref: 'fixes') }
before do
2018-11-18 11:00:15 +05:30
project.add_maintainer(user)
2018-11-08 19:23:39 +05:30
create(:merge_request,
source_project: forked_project,
target_project: project,
source_branch: 'fixes',
allow_collaboration: true)
2022-10-11 01:57:18 +05:30
create_build('rspec 1', :failed, test_stage)
2018-11-08 19:23:39 +05:30
end
it 'allows to retry failed pipeline' do
2019-02-15 15:39:39 +05:30
allow_any_instance_of(Project).to receive(:branch_allows_collaboration?).and_return(true)
2018-11-08 19:23:39 +05:30
allow_any_instance_of(Project).to receive(:empty_repo?).and_return(false)
service.execute(pipeline)
expect(build('rspec 1')).to be_pending
expect(pipeline.reload).to be_running
end
end
2017-08-17 22:00:37 +05:30
def statuses
pipeline.reload.statuses
end
2021-11-18 22:05:49 +05:30
# The method name can be confusing because this can actually return both Ci::Build and Ci::Bridge
2017-08-17 22:00:37 +05:30
def build(name)
statuses.latest.find_by(name: name)
end
2022-10-11 01:57:18 +05:30
def create_build(name, status, stage, **opts)
create_processable(:ci_build, name, status, stage, **opts)
2021-11-18 22:05:49 +05:30
end
2022-10-11 01:57:18 +05:30
def create_bridge(name, status, stage, **opts)
create_processable(:ci_bridge, name, status, stage, **opts)
2021-11-18 22:05:49 +05:30
end
2022-10-11 01:57:18 +05:30
def create_processable(type, name, status, stage, **opts)
2023-07-09 08:55:56 +05:30
create(
type,
name: name,
status: status,
ci_stage: stage,
stage_idx: stage.position,
pipeline: pipeline,
**opts
) do |_job|
2020-10-24 23:57:45 +05:30
::Ci::ProcessPipelineService.new(pipeline).execute
2017-08-17 22:00:37 +05:30
end
end
end