gitlab-org--gitlab-foss/spec/services/ci/register_job_service_spec.rb

337 lines
11 KiB
Ruby
Raw Normal View History

2015-09-15 20:23:43 +00:00
require 'spec_helper'
2015-09-15 13:00:45 +00:00
module Ci
describe RegisterJobService do
let!(:project) { FactoryGirl.create :project, shared_runners_enabled: false }
let!(:pipeline) { FactoryGirl.create :ci_pipeline, project: project }
let!(:pending_job) { FactoryGirl.create :ci_build, pipeline: pipeline }
2015-09-15 13:00:45 +00:00
let!(:shared_runner) { FactoryGirl.create(:ci_runner, is_shared: true) }
let!(:specific_runner) { FactoryGirl.create(:ci_runner, is_shared: false) }
before do
2015-12-04 11:55:23 +00:00
specific_runner.assign_to(project)
2015-08-26 01:42:46 +00:00
end
2016-07-11 22:12:31 +00:00
describe '#execute' do
2015-09-15 13:00:45 +00:00
context 'runner follow tag list' do
it "picks build with the same tag" do
pending_job.update(tag_list: ["linux"])
specific_runner.update(tag_list: ["linux"])
expect(execute(specific_runner)).to eq(pending_job)
2015-09-15 13:00:45 +00:00
end
it "does not pick build with different tag" do
pending_job.update(tag_list: ["linux"])
specific_runner.update(tag_list: ["win32"])
expect(execute(specific_runner)).to be_falsey
2015-09-15 13:00:45 +00:00
end
it "picks build without tag" do
expect(execute(specific_runner)).to eq(pending_job)
2015-09-15 13:00:45 +00:00
end
it "does not pick build with tag" do
pending_job.update(tag_list: ["linux"])
expect(execute(specific_runner)).to be_falsey
2015-09-15 13:00:45 +00:00
end
it "pick build without tag" do
specific_runner.update(tag_list: ["win32"])
expect(execute(specific_runner)).to eq(pending_job)
2015-09-15 13:00:45 +00:00
end
2015-08-26 01:42:46 +00:00
end
context 'deleted projects' do
before do
project.update(pending_delete: true)
end
context 'for shared runners' do
before do
project.update(shared_runners_enabled: true)
end
it 'does not pick a build' do
expect(execute(shared_runner)).to be_nil
end
end
context 'for specific runner' do
it 'does not pick a build' do
expect(execute(specific_runner)).to be_nil
end
end
end
2015-09-15 13:00:45 +00:00
context 'allow shared runners' do
before do
2015-12-04 11:55:23 +00:00
project.update(shared_runners_enabled: true)
2015-09-15 13:00:45 +00:00
end
2016-06-13 19:52:41 +00:00
context 'for multiple builds' do
let!(:project2) { create :project, shared_runners_enabled: true }
2016-06-13 19:52:41 +00:00
let!(:pipeline2) { create :ci_pipeline, project: project2 }
let!(:project3) { create :project, shared_runners_enabled: true }
2016-06-13 19:52:41 +00:00
let!(:pipeline3) { create :ci_pipeline, project: project3 }
let!(:build1_project1) { pending_job }
2016-06-13 19:52:41 +00:00
let!(:build2_project1) { FactoryGirl.create :ci_build, pipeline: pipeline }
let!(:build3_project1) { FactoryGirl.create :ci_build, pipeline: pipeline }
let!(:build1_project2) { FactoryGirl.create :ci_build, pipeline: pipeline2 }
let!(:build2_project2) { FactoryGirl.create :ci_build, pipeline: pipeline2 }
let!(:build1_project3) { FactoryGirl.create :ci_build, pipeline: pipeline3 }
it 'prefers projects without builds first' do
# it gets for one build from each of the projects
expect(execute(shared_runner)).to eq(build1_project1)
expect(execute(shared_runner)).to eq(build1_project2)
expect(execute(shared_runner)).to eq(build1_project3)
2016-06-13 19:52:41 +00:00
# then it gets a second build from each of the projects
expect(execute(shared_runner)).to eq(build2_project1)
expect(execute(shared_runner)).to eq(build2_project2)
2016-06-13 19:52:41 +00:00
# in the end the third build
expect(execute(shared_runner)).to eq(build3_project1)
2016-06-13 19:52:41 +00:00
end
it 'equalises number of running builds' do
# after finishing the first build for project 1, get a second build from the same project
expect(execute(shared_runner)).to eq(build1_project1)
2016-10-26 09:34:40 +00:00
build1_project1.reload.success
expect(execute(shared_runner)).to eq(build2_project1)
2016-06-13 19:52:41 +00:00
expect(execute(shared_runner)).to eq(build1_project2)
2016-10-26 09:34:40 +00:00
build1_project2.reload.success
expect(execute(shared_runner)).to eq(build2_project2)
expect(execute(shared_runner)).to eq(build1_project3)
expect(execute(shared_runner)).to eq(build3_project1)
2016-06-13 19:52:41 +00:00
end
end
2015-09-15 13:00:45 +00:00
context 'shared runner' do
let(:build) { execute(shared_runner) }
2015-09-15 13:00:45 +00:00
it { expect(build).to be_kind_of(Build) }
it { expect(build).to be_valid }
it { expect(build).to be_running }
it { expect(build.runner).to eq(shared_runner) }
end
context 'specific runner' do
let(:build) { execute(specific_runner) }
2015-09-15 13:00:45 +00:00
it { expect(build).to be_kind_of(Build) }
it { expect(build).to be_valid }
it { expect(build).to be_running }
it { expect(build.runner).to eq(specific_runner) }
end
2015-08-26 01:42:46 +00:00
end
2015-09-15 13:00:45 +00:00
context 'disallow shared runners' do
2015-11-05 15:22:37 +00:00
before do
2015-12-04 11:55:23 +00:00
project.update(shared_runners_enabled: false)
2015-11-05 15:22:37 +00:00
end
2015-09-15 13:00:45 +00:00
context 'shared runner' do
let(:build) { execute(shared_runner) }
2015-08-26 01:42:46 +00:00
2015-09-15 13:00:45 +00:00
it { expect(build).to be_nil }
end
2015-08-26 01:42:46 +00:00
2015-09-15 13:00:45 +00:00
context 'specific runner' do
let(:build) { execute(specific_runner) }
2015-08-26 01:42:46 +00:00
2015-09-15 13:00:45 +00:00
it { expect(build).to be_kind_of(Build) }
it { expect(build).to be_valid }
it { expect(build).to be_running }
it { expect(build.runner).to eq(specific_runner) }
end
2015-08-26 01:42:46 +00:00
end
2016-08-01 22:31:21 +00:00
context 'disallow when builds are disabled' do
before do
project.update(shared_runners_enabled: true)
project.project_feature.update_attribute(:builds_access_level, ProjectFeature::DISABLED)
end
context 'and uses shared runner' do
let(:build) { execute(shared_runner) }
2016-08-01 22:31:21 +00:00
it { expect(build).to be_nil }
end
context 'and uses specific runner' do
let(:build) { execute(specific_runner) }
2016-08-01 22:31:21 +00:00
it { expect(build).to be_nil }
end
end
context 'when first build is stalled' do
before do
pending_job.update(lock_version: 0)
end
subject { described_class.new(specific_runner).execute }
context 'with multiple builds are in queue' do
let!(:other_build) { create :ci_build, pipeline: pipeline }
before do
2017-03-06 10:56:44 +00:00
allow_any_instance_of(Ci::RegisterJobService).to receive(:builds_for_specific_runner)
.and_return(Ci::Build.where(id: [pending_job, other_build]))
end
it "receives second build from the queue" do
expect(subject).to be_valid
expect(subject.build).to eq(other_build)
end
end
context 'when single build is in queue' do
before do
2017-03-06 10:56:44 +00:00
allow_any_instance_of(Ci::RegisterJobService).to receive(:builds_for_specific_runner)
.and_return(Ci::Build.where(id: pending_job))
end
it "does not receive any valid result" do
expect(subject).not_to be_valid
end
end
context 'when there is no build in queue' do
before do
2017-03-06 10:56:44 +00:00
allow_any_instance_of(Ci::RegisterJobService).to receive(:builds_for_specific_runner)
.and_return(Ci::Build.none)
end
it "does not receive builds but result is valid" do
expect(subject).to be_valid
expect(subject.build).to be_nil
end
end
end
context 'when access_level of runner is not_protected' do
2017-09-02 08:53:54 +00:00
let!(:specific_runner) { create(:ci_runner, :specific) }
2017-08-22 08:01:11 +00:00
context 'when a job is protected' do
let!(:pending_job) { create(:ci_build, :protected, pipeline: pipeline) }
2017-08-22 08:01:11 +00:00
it 'picks the job' do
expect(execute(specific_runner)).to eq(pending_job)
2017-08-22 08:01:11 +00:00
end
end
context 'when a job is unprotected' do
let!(:pending_job) { create(:ci_build, pipeline: pipeline) }
2017-08-22 08:01:11 +00:00
it 'picks the job' do
expect(execute(specific_runner)).to eq(pending_job)
2017-09-02 07:31:14 +00:00
end
end
context 'when protected attribute of a job is nil' do
let!(:pending_job) { create(:ci_build, pipeline: pipeline) }
2017-09-02 07:31:14 +00:00
before do
pending_job.update_attribute(:protected, nil)
2017-09-02 07:31:14 +00:00
end
it 'picks the job' do
expect(execute(specific_runner)).to eq(pending_job)
2017-08-22 08:01:11 +00:00
end
end
end
context 'when access_level of runner is ref_protected' do
2017-08-29 08:31:59 +00:00
let!(:specific_runner) { create(:ci_runner, :ref_protected, :specific) }
2017-08-22 08:01:11 +00:00
context 'when a job is protected' do
let!(:pending_job) { create(:ci_build, :protected, pipeline: pipeline) }
2017-08-22 08:01:11 +00:00
it 'picks the job' do
expect(execute(specific_runner)).to eq(pending_job)
2017-08-22 08:01:11 +00:00
end
end
context 'when a job is unprotected' do
let!(:pending_job) { create(:ci_build, pipeline: pipeline) }
2017-08-22 08:01:11 +00:00
it 'does not pick the job' do
2017-08-22 08:01:11 +00:00
expect(execute(specific_runner)).to be_nil
end
end
2017-09-02 07:31:14 +00:00
context 'when protected attribute of a job is nil' do
let!(:pending_job) { create(:ci_build, pipeline: pipeline) }
2017-09-02 07:31:14 +00:00
before do
pending_job.update_attribute(:protected, nil)
2017-09-02 07:31:14 +00:00
end
it 'does not pick the job' do
2017-09-02 07:31:14 +00:00
expect(execute(specific_runner)).to be_nil
end
end
2017-08-22 08:01:11 +00:00
end
2017-09-03 14:35:37 +00:00
context 'when "dependencies" keyword is specified' do
2017-12-06 07:40:56 +00:00
before do
stub_feature_flags(ci_disable_validates_dependencies: false)
2017-12-06 07:40:56 +00:00
end
2017-09-03 14:35:37 +00:00
let!(:pre_stage_job) { create(:ci_build, :success, pipeline: pipeline, name: job_name, stage_idx: 0) }
let!(:pending_job) do
create(:ci_build, :pending, pipeline: pipeline, stage_idx: 1, options: { dependencies: ['spec'] } )
end
let(:picked_job) { execute(specific_runner) }
context 'when a depended job exists' do
let(:job_name) { 'spec' }
it "picks the build" do
expect(picked_job).to eq(pending_job)
end
context 'when "artifacts" keyword is specified on depended job' do
let!(:pre_stage_job) do
2017-09-05 15:51:05 +00:00
create(:ci_build,
:success,
:artifacts,
pipeline: pipeline,
name: job_name,
stage_idx: 0,
options: { artifacts: { paths: ['binaries/'] } } )
end
context 'when artifacts of depended job has existsed' do
it "picks the build" do
expect(picked_job).to eq(pending_job)
end
end
context 'when artifacts of depended job has not existsed' do
before do
2017-12-06 07:40:56 +00:00
pre_stage_job.erase
end
it 'does not pick the build and drops the build' do
expect(picked_job).to be_nil
expect(pending_job.reload).to be_failed
expect(pending_job).to be_missing_dependency_failure
end
end
end
2017-09-03 14:35:37 +00:00
end
end
def execute(runner)
described_class.new(runner).execute.build
end
2015-08-26 01:42:46 +00:00
end
end
end