Commit 3e706636 authored by Kamil Trzciński's avatar Kamil Trzciński

Fetch build one-by-one

This is based on hypothesis:

- Long queues inflate memory allocation
- Long queues make us go very deep into a queue
  as that build could already be picked by another
  runner, thus increasing amount of retries
parent e294fa99
...@@ -9,6 +9,7 @@ module Ci ...@@ -9,6 +9,7 @@ module Ci
include FromUnion include FromUnion
include TokenAuthenticatable include TokenAuthenticatable
include IgnorableColumns include IgnorableColumns
include FeatureGate
add_authentication_token_field :token, encrypted: -> { Feature.enabled?(:ci_runners_tokens_optional_encryption, default_enabled: true) ? :optional : :required } add_authentication_token_field :token, encrypted: -> { Feature.enabled?(:ci_runners_tokens_optional_encryption, default_enabled: true) ? :optional : :required }
......
...@@ -23,36 +23,11 @@ module Ci ...@@ -23,36 +23,11 @@ module Ci
private private
# rubocop: disable CodeReuse/ActiveRecord
def process_queue(params) def process_queue(params)
builds =
if runner.instance_type?
builds_for_shared_runner
elsif runner.group_type?
builds_for_group_runner
else
builds_for_project_runner
end
# pick builds that does not have other tags than runner's one
builds = builds.matches_tag_ids(runner.tags.ids)
# pick builds that have at least one tag
unless runner.run_untagged?
builds = builds.with_any_tags
end
# pick builds that older than specified age
if params.key?(:job_age)
builds = builds.queued_before(params[:job_age].seconds.ago)
end
@metrics.observe_queue_size(-> { builds.to_a.size })
valid = true valid = true
depth = 0 depth = 0
builds.each do |build| each_build(params) do |build|
depth += 1 depth += 1
@metrics.increment_queue_operation(:queue_iteration) @metrics.increment_queue_operation(:queue_iteration)
...@@ -78,9 +53,53 @@ module Ci ...@@ -78,9 +53,53 @@ module Ci
Result.new(nil, nil, valid) Result.new(nil, nil, valid)
end end
# rubocop: disable CodeReuse/ActiveRecord
def each_build(params, &blk)
builds =
if runner.instance_type?
builds_for_shared_runner
elsif runner.group_type?
builds_for_group_runner
else
builds_for_project_runner
end
# pick builds that does not have other tags than runner's one
builds = builds.matches_tag_ids(runner.tags.ids)
# pick builds that have at least one tag
unless runner.run_untagged?
builds = builds.with_any_tags
end
# pick builds that older than specified age
if params.key?(:job_age)
builds = builds.queued_before(params[:job_age].seconds.ago)
end
if Feature.enabled?(:ci_register_job_service_one_by_one, runner)
build_ids = builds.pluck(:id)
@metrics.observe_queue_size(-> { build_ids.size })
build_ids.each do |build_id|
yield Ci::Build.find(build_id)
end
else
@metrics.observe_queue_size(-> { builds.to_a.size })
builds.each(&blk)
end
end
# rubocop: enable CodeReuse/ActiveRecord # rubocop: enable CodeReuse/ActiveRecord
def process_build(build, params) def process_build(build, params)
unless build.pending?
@metrics.increment_queue_operation(:build_not_pending)
return
end
if runner.can_pick?(build) if runner.can_pick?(build)
@metrics.increment_queue_operation(:build_can_pick) @metrics.increment_queue_operation(:build_can_pick)
else else
......
---
name: ci_register_job_service_one_by_one
introduced_by_url: https://gitlab.com/gitlab-org/gitlab/-/merge_requests/55194
rollout_issue_url: https://gitlab.com/gitlab-org/gitlab/-/issues/323177
milestone: '13.10'
type: development
group: group::memory
default_enabled: false
...@@ -19,6 +19,7 @@ module Gitlab ...@@ -19,6 +19,7 @@ module Gitlab
OPERATION_COUNTERS = [ OPERATION_COUNTERS = [
:build_can_pick, :build_can_pick,
:build_not_pick, :build_not_pick,
:build_not_pending,
:build_conflict_lock, :build_conflict_lock,
:build_conflict_exception, :build_conflict_exception,
:build_conflict_transition, :build_conflict_transition,
......
...@@ -13,573 +13,615 @@ module Ci ...@@ -13,573 +13,615 @@ module Ci
let!(:pending_job) { create(:ci_build, pipeline: pipeline) } let!(:pending_job) { create(:ci_build, pipeline: pipeline) }
describe '#execute' do describe '#execute' do
context 'runner follow tag list' do shared_examples 'handles runner assignment' do
it "picks build with the same tag" do context 'runner follow tag list' do
pending_job.update!(tag_list: ["linux"]) it "picks build with the same tag" do
specific_runner.update!(tag_list: ["linux"]) pending_job.update!(tag_list: ["linux"])
expect(execute(specific_runner)).to eq(pending_job) specific_runner.update!(tag_list: ["linux"])
end expect(execute(specific_runner)).to eq(pending_job)
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
end
it "picks build without tag" do it "does not pick build with different tag" do
expect(execute(specific_runner)).to eq(pending_job) pending_job.update!(tag_list: ["linux"])
end specific_runner.update!(tag_list: ["win32"])
expect(execute(specific_runner)).to be_falsey
end
it "does not pick build with tag" do it "picks build without tag" do
pending_job.update!(tag_list: ["linux"]) expect(execute(specific_runner)).to eq(pending_job)
expect(execute(specific_runner)).to be_falsey end
end
it "pick build without tag" do it "does not pick build with tag" do
specific_runner.update!(tag_list: ["win32"]) pending_job.update!(tag_list: ["linux"])
expect(execute(specific_runner)).to eq(pending_job) expect(execute(specific_runner)).to be_falsey
end end
end
context 'deleted projects' do it "pick build without tag" do
before do specific_runner.update!(tag_list: ["win32"])
project.update!(pending_delete: true) expect(execute(specific_runner)).to eq(pending_job)
end
end end
context 'for shared runners' do context 'deleted projects' do
before do before do
project.update!(shared_runners_enabled: true) project.update!(pending_delete: true)
end end
it 'does not pick a build' do context 'for shared runners' do
expect(execute(shared_runner)).to be_nil 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 end
end
context 'for specific runner' do context 'for specific runner' do
it 'does not pick a build' do it 'does not pick a build' do
expect(execute(specific_runner)).to be_nil expect(execute(specific_runner)).to be_nil
end
end end
end end
end
context 'allow shared runners' do context 'allow shared runners' do
before do before do
project.update!(shared_runners_enabled: true) project.update!(shared_runners_enabled: true)
end end
context 'for multiple builds' do context 'for multiple builds' do
let!(:project2) { create :project, shared_runners_enabled: true } let!(:project2) { create :project, shared_runners_enabled: true }
let!(:pipeline2) { create :ci_pipeline, project: project2 } let!(:pipeline2) { create :ci_pipeline, project: project2 }
let!(:project3) { create :project, shared_runners_enabled: true } let!(:project3) { create :project, shared_runners_enabled: true }
let!(:pipeline3) { create :ci_pipeline, project: project3 } let!(:pipeline3) { create :ci_pipeline, project: project3 }
let!(:build1_project1) { pending_job } let!(:build1_project1) { pending_job }
let!(:build2_project1) { FactoryBot.create :ci_build, pipeline: pipeline } let!(:build2_project1) { FactoryBot.create :ci_build, pipeline: pipeline }
let!(:build3_project1) { FactoryBot.create :ci_build, pipeline: pipeline } let!(:build3_project1) { FactoryBot.create :ci_build, pipeline: pipeline }
let!(:build1_project2) { FactoryBot.create :ci_build, pipeline: pipeline2 } let!(:build1_project2) { FactoryBot.create :ci_build, pipeline: pipeline2 }
let!(:build2_project2) { FactoryBot.create :ci_build, pipeline: pipeline2 } let!(:build2_project2) { FactoryBot.create :ci_build, pipeline: pipeline2 }
let!(:build1_project3) { FactoryBot.create :ci_build, pipeline: pipeline3 } let!(:build1_project3) { FactoryBot.create :ci_build, pipeline: pipeline3 }
it 'prefers projects without builds first' do it 'prefers projects without builds first' do
# it gets for one build from each of the projects # 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_project1)
expect(execute(shared_runner)).to eq(build1_project2) expect(execute(shared_runner)).to eq(build1_project2)
expect(execute(shared_runner)).to eq(build1_project3) expect(execute(shared_runner)).to eq(build1_project3)
# then it gets a second build from each of the projects # 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_project1)
expect(execute(shared_runner)).to eq(build2_project2) expect(execute(shared_runner)).to eq(build2_project2)
# in the end the third build # in the end the third build
expect(execute(shared_runner)).to eq(build3_project1) expect(execute(shared_runner)).to eq(build3_project1)
end end
it 'equalises number of running builds' do it 'equalises number of running builds' do
# after finishing the first build for project 1, get a second build from the same project # after finishing the first build for project 1, get a second build from the same project
expect(execute(shared_runner)).to eq(build1_project1) expect(execute(shared_runner)).to eq(build1_project1)
build1_project1.reload.success build1_project1.reload.success
expect(execute(shared_runner)).to eq(build2_project1) expect(execute(shared_runner)).to eq(build2_project1)
expect(execute(shared_runner)).to eq(build1_project2) expect(execute(shared_runner)).to eq(build1_project2)
build1_project2.reload.success build1_project2.reload.success
expect(execute(shared_runner)).to eq(build2_project2) expect(execute(shared_runner)).to eq(build2_project2)
expect(execute(shared_runner)).to eq(build1_project3) expect(execute(shared_runner)).to eq(build1_project3)
expect(execute(shared_runner)).to eq(build3_project1) expect(execute(shared_runner)).to eq(build3_project1)
end
end end
end
context 'shared runner' do context 'shared runner' do
let(:response) { described_class.new(shared_runner).execute } let(:response) { described_class.new(shared_runner).execute }
let(:build) { response.build } let(:build) { response.build }
it { expect(build).to be_kind_of(Build) } it { expect(build).to be_kind_of(Build) }
it { expect(build).to be_valid } it { expect(build).to be_valid }
it { expect(build).to be_running } it { expect(build).to be_running }
it { expect(build.runner).to eq(shared_runner) } it { expect(build.runner).to eq(shared_runner) }
it { expect(Gitlab::Json.parse(response.build_json)['id']).to eq(build.id) } it { expect(Gitlab::Json.parse(response.build_json)['id']).to eq(build.id) }
end end
context 'specific runner' do context 'specific runner' do
let(:build) { execute(specific_runner) } let(:build) { execute(specific_runner) }
it { expect(build).to be_kind_of(Build) } it { expect(build).to be_kind_of(Build) }
it { expect(build).to be_valid } it { expect(build).to be_valid }
it { expect(build).to be_running } it { expect(build).to be_running }
it { expect(build.runner).to eq(specific_runner) } it { expect(build.runner).to eq(specific_runner) }
end
end end
end
context 'disallow shared runners' do context 'disallow shared runners' do
before do before do
project.update!(shared_runners_enabled: false) project.update!(shared_runners_enabled: false)
end end
context 'shared runner' do context 'shared runner' do
let(:build) { execute(shared_runner) } let(:build) { execute(shared_runner) }
it { expect(build).to be_nil } it { expect(build).to be_nil }
end end
context 'specific runner' do context 'specific runner' do
let(:build) { execute(specific_runner) } let(:build) { execute(specific_runner) }
it { expect(build).to be_kind_of(Build) } it { expect(build).to be_kind_of(Build) }
it { expect(build).to be_valid } it { expect(build).to be_valid }
it { expect(build).to be_running } it { expect(build).to be_running }
it { expect(build.runner).to eq(specific_runner) } it { expect(build.runner).to eq(specific_runner) }
end
end end
end
context 'disallow when builds are disabled' do context 'disallow when builds are disabled' do
before do before do
project.update!(shared_runners_enabled: true, group_runners_enabled: true) project.update!(shared_runners_enabled: true, group_runners_enabled: true)
project.project_feature.update_attribute(:builds_access_level, ProjectFeature::DISABLED) project.project_feature.update_attribute(:builds_access_level, ProjectFeature::DISABLED)
end end
context 'and uses shared runner' do context 'and uses shared runner' do
let(:build) { execute(shared_runner) } let(:build) { execute(shared_runner) }
it { expect(build).to be_nil } it { expect(build).to be_nil }
end end
context 'and uses group runner' do context 'and uses group runner' do
let(:build) { execute(group_runner) } let(:build) { execute(group_runner) }
it { expect(build).to be_nil } it { expect(build).to be_nil }
end end
context 'and uses project runner' do context 'and uses project runner' do
let(:build) { execute(specific_runner) } let(:build) { execute(specific_runner) }
it { expect(build).to be_nil } it { expect(build).to be_nil }
end
end end
end
context 'allow group runners' do context 'allow group runners' do
before do before do
project.update!(group_runners_enabled: true) project.update!(group_runners_enabled: true)
end end
context 'for multiple builds' do context 'for multiple builds' do
let!(:project2) { create(:project, group_runners_enabled: true, group: group) } let!(:project2) { create(:project, group_runners_enabled: true, group: group) }
let!(:pipeline2) { create(:ci_pipeline, project: project2) } let!(:pipeline2) { create(:ci_pipeline, project: project2) }
let!(:project3) { create(:project, group_runners_enabled: true, group: group) } let!(:project3) { create(:project, group_runners_enabled: true, group: group) }
let!(:pipeline3) { create(:ci_pipeline, project: project3) } let!(:pipeline3) { create(:ci_pipeline, project: project3) }
let!(:build1_project1) { pending_job } let!(:build1_project1) { pending_job }
let!(:build2_project1) { create(:ci_build, pipeline: pipeline) } let!(:build2_project1) { create(:ci_build, pipeline: pipeline) }
let!(:build3_project1) { create(:ci_build, pipeline: pipeline) } let!(:build3_project1) { create(:ci_build, pipeline: pipeline) }
let!(:build1_project2) { create(:ci_build, pipeline: pipeline2) } let!(:build1_project2) { create(:ci_build, pipeline: pipeline2) }
let!(:build2_project2) { create(:ci_build, pipeline: pipeline2) } let!(:build2_project2) { create(:ci_build, pipeline: pipeline2) }
let!(:build1_project3) { create(:ci_build, pipeline: pipeline3) } let!(:build1_project3) { create(:ci_build, pipeline: pipeline3) }
# these shouldn't influence the scheduling # these shouldn't influence the scheduling
let!(:unrelated_group) { create(:group) } let!(:unrelated_group) { create(:group) }
let!(:unrelated_project) { create(:project, group_runners_enabled: true, group: unrelated_group) } let!(:unrelated_project) { create(:project, group_runners_enabled: true, group: unrelated_group) }
let!(:unrelated_pipeline) { create(:ci_pipeline, project: unrelated_project) } let!(:unrelated_pipeline) { create(:ci_pipeline, project: unrelated_project) }
let!(:build1_unrelated_project) { create(:ci_build, pipeline: unrelated_pipeline) } let!(:build1_unrelated_project) { create(:ci_build, pipeline: unrelated_pipeline) }
let!(:unrelated_group_runner) { create(:ci_runner, :group, groups: [unrelated_group]) } let!(:unrelated_group_runner) { create(:ci_runner, :group, groups: [unrelated_group]) }
it 'does not consider builds from other group runners' do it 'does not consider builds from other group runners' do
expect(described_class.new(group_runner).send(:builds_for_group_runner).count).to eq 6 expect(described_class.new(group_runner).send(:builds_for_group_runner).count).to eq 6
execute(group_runner) execute(group_runner)
expect(described_class.new(group_runner).send(:builds_for_group_runner).count).to eq 5 expect(described_class.new(group_runner).send(:builds_for_group_runner).count).to eq 5
execute(group_runner) execute(group_runner)
expect(described_class.new(group_runner).send(:builds_for_group_runner).count).to eq 4 expect(described_class.new(group_runner).send(:builds_for_group_runner).count).to eq 4
execute(group_runner) execute(group_runner)
expect(described_class.new(group_runner).send(:builds_for_group_runner).count).to eq 3 expect(described_class.new(group_runner).send(:builds_for_group_runner).count).to eq 3
execute(group_runner) execute(group_runner)
expect(described_class.new(group_runner).send(:builds_for_group_runner).count).to eq 2 expect(described_class.new(group_runner).send(:builds_for_group_runner).count).to eq 2
execute(group_runner) execute(group_runner)
expect(described_class.new(group_runner).send(:builds_for_group_runner).count).to eq 1 expect(described_class.new(group_runner).send(:builds_for_group_runner).count).to eq 1
execute(group_runner) execute(group_runner)
expect(described_class.new(group_runner).send(:builds_for_group_runner).count).to eq 0 expect(described_class.new(group_runner).send(:builds_for_group_runner).count).to eq 0
expect(execute(group_runner)).to be_nil expect(execute(group_runner)).to be_nil
end
end end
end
context 'group runner' do context 'group runner' do
let(:build) { execute(group_runner) } let(:build) { execute(group_runner) }
it { expect(build).to be_kind_of(Build) } it { expect(build).to be_kind_of(Build) }
it { expect(build).to be_valid } it { expect(build).to be_valid }
it { expect(build).to be_running } it { expect(build).to be_running }
it { expect(build.runner).to eq(group_runner) } it { expect(build.runner).to eq(group_runner) }
end
end end
end
context 'disallow group runners' do context 'disallow group runners' do
before do before do
project.update!(group_runners_enabled: false) project.update!(group_runners_enabled: false)
end end
context 'group runner' do context 'group runner' do
let(:build) { execute(group_runner) } let(:build) { execute(group_runner) }
it { expect(build).to be_nil } it { expect(build).to be_nil }
end
end end
end
context 'when first build is stalled' do context 'when first build is stalled' do
before do before do
allow_any_instance_of(Ci::RegisterJobService).to receive(:assign_runner!).and_call_original allow_any_instance_of(Ci::RegisterJobService).to receive(:assign_runner!).and_call_original
allow_any_instance_of(Ci::RegisterJobService).to receive(:assign_runner!) allow_any_instance_of(Ci::RegisterJobService).to receive(:assign_runner!)
.with(pending_job, anything).and_raise(ActiveRecord::StaleObjectError) .with(pending_job, anything).and_raise(ActiveRecord::StaleObjectError)
end end
subject { described_class.new(specific_runner).execute } subject { described_class.new(specific_runner).execute }
context 'with multiple builds are in queue' do context 'with multiple builds are in queue' do
let!(:other_build) { create :ci_build, pipeline: pipeline } let!(:other_build) { create :ci_build, pipeline: pipeline }
before do before do
allow_any_instance_of(Ci::RegisterJobService).to receive(:builds_for_project_runner) allow_any_instance_of(Ci::RegisterJobService).to receive(:builds_for_project_runner)
.and_return(Ci::Build.where(id: [pending_job, other_build])) .and_return(Ci::Build.where(id: [pending_job, other_build]))
end end
it "receives second build from the queue" do it "receives second build from the queue" do
expect(subject).to be_valid expect(subject).to be_valid
expect(subject.build).to eq(other_build) expect(subject.build).to eq(other_build)
end
end end
end
context 'when single build is in queue' do context 'when single build is in queue' do
before do before do
allow_any_instance_of(Ci::RegisterJobService).to receive(:builds_for_project_runner) allow_any_instance_of(Ci::RegisterJobService).to receive(:builds_for_project_runner)
.and_return(Ci::Build.where(id: pending_job)) .and_return(Ci::Build.where(id: pending_job))
end end
it "does not receive any valid result" do it "does not receive any valid result" do
expect(subject).not_to be_valid expect(subject).not_to be_valid
end
end end
end
context 'when there is no build in queue' do context 'when there is no build in queue' do
before do before do
allow_any_instance_of(Ci::RegisterJobService).to receive(:builds_for_project_runner) allow_any_instance_of(Ci::RegisterJobService).to receive(:builds_for_project_runner)
.and_return(Ci::Build.none) .and_return(Ci::Build.none)
end end
it "does not receive builds but result is valid" do it "does not receive builds but result is valid" do
expect(subject).to be_valid expect(subject).to be_valid
expect(subject.build).to be_nil expect(subject.build).to be_nil
end
end end
end end
end
context 'when access_level of runner is not_protected' do context 'when access_level of runner is not_protected' do
let!(:specific_runner) { create(:ci_runner, :project, projects: [project]) } let!(:specific_runner) { create(:ci_runner, :project, projects: [project]) }
context 'when a job is protected' do context 'when a job is protected' do
let!(:pending_job) { create(:ci_build, :protected, pipeline: pipeline) } let!(:pending_job) { create(:ci_build, :protected, pipeline: pipeline) }
it 'picks the job' do it 'picks the job' do
expect(execute(specific_runner)).to eq(pending_job) expect(execute(specific_runner)).to eq(pending_job)
end
end end
end
context 'when a job is unprotected' do context 'when a job is unprotected' do
let!(:pending_job) { create(:ci_build, pipeline: pipeline) } let!(:pending_job) { create(:ci_build, pipeline: pipeline) }
it 'picks the job' do it 'picks the job' do
expect(execute(specific_runner)).to eq(pending_job) expect(execute(specific_runner)).to eq(pending_job)
end
end end
end
context 'when protected attribute of a job is nil' do context 'when protected attribute of a job is nil' do
let!(:pending_job) { create(:ci_build, pipeline: pipeline) } let!(:pending_job) { create(:ci_build, pipeline: pipeline) }
before do before do
pending_job.update_attribute(:protected, nil) pending_job.update_attribute(:protected, nil)
end end
it 'picks the job' do it 'picks the job' do
expect(execute(specific_runner)).to eq(pending_job) expect(execute(specific_runner)).to eq(pending_job)
end
end end
end end
end
context 'when access_level of runner is ref_protected' do context 'when access_level of runner is ref_protected' do
let!(:specific_runner) { create(:ci_runner, :project, :ref_protected, projects: [project]) } let!(:specific_runner) { create(:ci_runner, :project, :ref_protected, projects: [project]) }
context 'when a job is protected' do context 'when a job is protected' do
let!(:pending_job) { create(:ci_build, :protected, pipeline: pipeline) } let!(:pending_job) { create(:ci_build, :protected, pipeline: pipeline) }
it 'picks the job' do it 'picks the job' do
expect(execute(specific_runner)).to eq(pending_job) expect(execute(specific_runner)).to eq(pending_job)
end
end end
end
context 'when a job is unprotected' do context 'when a job is unprotected' do
let!(:pending_job) { create(:ci_build, pipeline: pipeline) } let!(:pending_job) { create(:ci_build, pipeline: pipeline) }
it 'does not pick the job' do it 'does not pick the job' do
expect(execute(specific_runner)).to be_nil expect(execute(specific_runner)).to be_nil
end
end end
end
context 'when protected attribute of a job is nil' do context 'when protected attribute of a job is nil' do
let!(:pending_job) { create(:ci_build, pipeline: pipeline) } let!(:pending_job) { create(:ci_build, pipeline: pipeline) }
before do before do
pending_job.update_attribute(:protected, nil) pending_job.update_attribute(:protected, nil)
end end
it 'does not pick the job' do it 'does not pick the job' do
expect(execute(specific_runner)).to be_nil expect(execute(specific_runner)).to be_nil
end
end end
end end
end
context 'runner feature set is verified' do context 'runner feature set is verified' do
let(:options) { { artifacts: { reports: { junit: "junit.xml" } } } } let(:options) { { artifacts: { reports: { junit: "junit.xml" } } } }
let!(:pending_job) { create(:ci_build, :pending, pipeline: pipeline, options: options) } let!(:pending_job) { create(:ci_build, :pending, pipeline: pipeline, options: options) }
subject { execute(specific_runner, params) } subject { execute(specific_runner, params) }
context 'when feature is missing by runner' do context 'when feature is missing by runner' do
let(:params) { {} } let(:params) { {} }
it 'does not pick the build and drops the build' do it 'does not pick the build and drops the build' do
expect(subject).to be_nil expect(subject).to be_nil
expect(pending_job.reload).to be_failed expect(pending_job.reload).to be_failed
expect(pending_job).to be_runner_unsupported expect(pending_job).to be_runner_unsupported
end end
end
context 'when feature is supported by runner' do
let(:params) do
{ info: { features: { upload_multiple_artifacts: true } } }
end end
it 'does pick job' do context 'when feature is supported by runner' do
expect(subject).not_to be_nil let(:params) do
end { info: { features: { upload_multiple_artifacts: true } } }
end end
end
context 'when "dependencies" keyword is specified' do it 'does pick job' do
shared_examples 'not pick' do expect(subject).not_to be_nil
it 'does not pick the build and drops the build' do end
expect(subject).to be_nil
expect(pending_job.reload).to be_failed
expect(pending_job).to be_missing_dependency_failure
end end
end end
shared_examples 'validation is active' do context 'when "dependencies" keyword is specified' do
context 'when depended job has not been completed yet' do shared_examples 'not pick' do
let!(:pre_stage_job) { create(:ci_build, :manual, pipeline: pipeline, name: 'test', stage_idx: 0) } it 'does not pick the build and drops the build' do
expect(subject).to be_nil
it { expect(subject).to eq(pending_job) } expect(pending_job.reload).to be_failed
expect(pending_job).to be_missing_dependency_failure
end
end end
context 'when artifacts of depended job has been expired' do shared_examples 'validation is active' do
let!(:pre_stage_job) { create(:ci_build, :success, :expired, pipeline: pipeline, name: 'test', stage_idx: 0) } context 'when depended job has not been completed yet' do
let!(:pre_stage_job) { create(:ci_build, :manual, pipeline: pipeline, name: 'test', stage_idx: 0) }
it_behaves_like 'not pick' it { expect(subject).to eq(pending_job) }
end end
context 'when artifacts of depended job has been erased' do context 'when artifacts of depended job has been expired' do
let!(:pre_stage_job) { create(:ci_build, :success, pipeline: pipeline, name: 'test', stage_idx: 0, erased_at: 1.minute.ago) } let!(:pre_stage_job) { create(:ci_build, :success, :expired, pipeline: pipeline, name: 'test', stage_idx: 0) }
before do it_behaves_like 'not pick'
pre_stage_job.erase
end end
it_behaves_like 'not pick' context 'when artifacts of depended job has been erased' do
end let!(:pre_stage_job) { create(:ci_build, :success, pipeline: pipeline, name: 'test', stage_idx: 0, erased_at: 1.minute.ago) }
context 'when job object is staled' do
let!(:pre_stage_job) { create(:ci_build, :success, :expired, pipeline: pipeline, name: 'test', stage_idx: 0) }
before do before do
allow_any_instance_of(Ci::Build).to receive(:drop!) pre_stage_job.erase
.and_raise(ActiveRecord::StaleObjectError.new(pending_job, :drop!)) end
end
it 'does not drop nor pick' do it_behaves_like 'not pick'
expect(subject).to be_nil
end end
end
end
shared_examples 'validation is not active' do context 'when job object is staled' do
context 'when depended job has not been completed yet' do let!(:pre_stage_job) { create(:ci_build, :success, :expired, pipeline: pipeline, name: 'test', stage_idx: 0) }
let!(:pre_stage_job) { create(:ci_build, :manual, pipeline: pipeline, name: 'test', stage_idx: 0) }
it { expect(subject).to eq(pending_job) }
end
context 'when artifacts of depended job has been expired' do before do
let!(:pre_stage_job) { create(:ci_build, :success, :expired, pipeline: pipeline, name: 'test', stage_idx: 0) } allow_any_instance_of(Ci::Build).to receive(:drop!)
.and_raise(ActiveRecord::StaleObjectError.new(pending_job, :drop!))
end
it { expect(subject).to eq(pending_job) } it 'does not drop nor pick' do
expect(subject).to be_nil
end
end
end end
context 'when artifacts of depended job has been erased' do shared_examples 'validation is not active' do
let!(:pre_stage_job) { create(:ci_build, :success, pipeline: pipeline, name: 'test', stage_idx: 0, erased_at: 1.minute.ago) } context 'when depended job has not been completed yet' do
let!(:pre_stage_job) { create(:ci_build, :manual, pipeline: pipeline, name: 'test', stage_idx: 0) }
before do it { expect(subject).to eq(pending_job) }
pre_stage_job.erase
end end
it { expect(subject).to eq(pending_job) } context 'when artifacts of depended job has been expired' do
end let!(:pre_stage_job) { create(:ci_build, :success, :expired, pipeline: pipeline, name: 'test', stage_idx: 0) }
end
before do it { expect(subject).to eq(pending_job) }
stub_feature_flags(ci_validate_build_dependencies_override: false) end
end
let!(:pre_stage_job) { create(:ci_build, :success, pipeline: pipeline, name: 'test', stage_idx: 0) } context 'when artifacts of depended job has been erased' do
let!(:pre_stage_job) { create(:ci_build, :success, pipeline: pipeline, name: 'test', stage_idx: 0, erased_at: 1.minute.ago) }
let!(:pending_job) do before do
create(:ci_build, :pending, pre_stage_job.erase
pipeline: pipeline, stage_idx: 1, end
options: { script: ["bash"], dependencies: ['test'] })
end
subject { execute(specific_runner) } it { expect(subject).to eq(pending_job) }
end
end
context 'when validates for dependencies is enabled' do
before do before do
stub_feature_flags(ci_validate_build_dependencies_override: false) stub_feature_flags(ci_validate_build_dependencies_override: false)
end end
it_behaves_like 'validation is active' let!(:pre_stage_job) { create(:ci_build, :success, pipeline: pipeline, name: 'test', stage_idx: 0) }
let!(:pending_job) do
create(:ci_build, :pending,
pipeline: pipeline, stage_idx: 1,
options: { script: ["bash"], dependencies: ['test'] })
end
subject { execute(specific_runner) }
context 'when the main feature flag is enabled for a specific project' do context 'when validates for dependencies is enabled' do
before do before do
stub_feature_flags(ci_validate_build_dependencies: pipeline.project) stub_feature_flags(ci_validate_build_dependencies_override: false)
end end
it_behaves_like 'validation is active' it_behaves_like 'validation is active'
context 'when the main feature flag is enabled for a specific project' do
before do
stub_feature_flags(ci_validate_build_dependencies: pipeline.project)
end
it_behaves_like 'validation is active'
end
context 'when the main feature flag is enabled for a different project' do
before do
stub_feature_flags(ci_validate_build_dependencies: create(:project))
end
it_behaves_like 'validation is not active'
end
end end
context 'when the main feature flag is enabled for a different project' do context 'when validates for dependencies is disabled' do
before do before do
stub_feature_flags(ci_validate_build_dependencies: create(:project)) stub_feature_flags(ci_validate_build_dependencies_override: true)
end end
it_behaves_like 'validation is not active' it_behaves_like 'validation is not active'
end end
end end
context 'when validates for dependencies is disabled' do context 'when build is degenerated' do
before do let!(:pending_job) { create(:ci_build, :pending, :degenerated, pipeline: pipeline) }
stub_feature_flags(ci_validate_build_dependencies_override: true)
end subject { execute(specific_runner, {}) }
it_behaves_like 'validation is not active' it 'does not pick the build and drops the build' do
expect(subject).to be_nil
pending_job.reload
expect(pending_job).to be_failed
expect(pending_job).to be_archived_failure
end
end end
end
context 'when build is degenerated' do context 'when build has data integrity problem' do
let!(:pending_job) { create(:ci_build, :pending, :degenerated, pipeline: pipeline) } let!(:pending_job) do
create(:ci_build, :pending, pipeline: pipeline)
end
subject { execute(specific_runner, {}) } before do
pending_job.update_columns(options: "string")
end
it 'does not pick the build and drops the build' do subject { execute(specific_runner, {}) }
expect(subject).to be_nil
pending_job.reload it 'does drop the build and logs both failures' do
expect(pending_job).to be_failed expect(Gitlab::ErrorTracking).to receive(:track_exception)
expect(pending_job).to be_archived_failure .with(anything, a_hash_including(build_id: pending_job.id))
end .twice
end .and_call_original
context 'when build has data integrity problem' do expect(subject).to be_nil
let!(:pending_job) do
create(:ci_build, :pending, pipeline: pipeline)
end
before do pending_job.reload
pending_job.update_columns(options: "string") expect(pending_job).to be_failed
expect(pending_job).to be_data_integrity_failure
end
end end
subject { execute(specific_runner, {}) } context 'when build fails to be run!' do
let!(:pending_job) do
create(:ci_build, :pending, pipeline: pipeline)
end
it 'does drop the build and logs both failures' do before do
expect(Gitlab::ErrorTracking).to receive(:track_exception) expect_any_instance_of(Ci::Build).to receive(:run!)
.with(anything, a_hash_including(build_id: pending_job.id)) .and_raise(RuntimeError, 'scheduler error')
.twice end
.and_call_original
expect(subject).to be_nil subject { execute(specific_runner, {}) }
pending_job.reload it 'does drop the build and logs failure' do
expect(pending_job).to be_failed expect(Gitlab::ErrorTracking).to receive(:track_exception)
expect(pending_job).to be_data_integrity_failure .with(anything, a_hash_including(build_id: pending_job.id))
end .once
end .and_call_original
context 'when build fails to be run!' do expect(subject).to be_nil
let!(:pending_job) do
create(:ci_build, :pending, pipeline: pipeline) pending_job.reload
expect(pending_job).to be_failed
expect(pending_job).to be_scheduler_failure
end
end end
before do context 'when an exception is raised during a persistent ref creation' do
expect_any_instance_of(Ci::Build).to receive(:run!) before do
.and_raise(RuntimeError, 'scheduler error') allow_any_instance_of(Ci::PersistentRef).to receive(:exist?) { false }
allow_any_instance_of(Ci::PersistentRef).to receive(:create_ref) { raise ArgumentError }
end
subject { execute(specific_runner, {}) }
it 'picks the build' do
expect(subject).to eq(pending_job)
pending_job.reload
expect(pending_job).to be_running
end
end end
subject { execute(specific_runner, {}) } context 'when only some builds can be matched by runner' do
let!(:specific_runner) { create(:ci_runner, :project, projects: [project], tag_list: %w[matching]) }
let!(:pending_job) { create(:ci_build, pipeline: pipeline, tag_list: %w[matching]) }
it 'does drop the build and logs failure' do before do
expect(Gitlab::ErrorTracking).to receive(:track_exception) # create additional matching and non-matching jobs
.with(anything, a_hash_including(build_id: pending_job.id)) create_list(:ci_build, 2, pipeline: pipeline, tag_list: %w[matching])
.once create(:ci_build, pipeline: pipeline, tag_list: %w[non-matching])
.and_call_original end
expect(subject).to be_nil it "observes queue size of only matching jobs" do
# pending_job + 2 x matching ones
expect(Gitlab::Ci::Queue::Metrics.queue_size_total).to receive(:observe).with({}, 3)
pending_job.reload expect(execute(specific_runner)).to eq(pending_job)
expect(pending_job).to be_failed end
expect(pending_job).to be_scheduler_failure
end end
end end
context 'when an exception is raised during a persistent ref creation' do context 'when ci_register_job_service_one_by_one is enabled' do
before do before do
allow_any_instance_of(Ci::PersistentRef).to receive(:exist?) { false } stub_feature_flags(ci_register_job_service_one_by_one: true)
allow_any_instance_of(Ci::PersistentRef).to receive(:create_ref) { raise ArgumentError }
end end
subject { execute(specific_runner, {}) } it 'picks builds one-by-one' do
expect(Ci::Build).to receive(:find).with(pending_job.id).and_call_original
it 'picks the build' do expect(execute(specific_runner)).to eq(pending_job)
expect(subject).to eq(pending_job) end
include_examples 'handles runner assignment'
end
pending_job.reload context 'when ci_register_job_service_one_by_one is disabled' do
expect(pending_job).to be_running before do
stub_feature_flags(ci_register_job_service_one_by_one: false)
end end
include_examples 'handles runner assignment'
end end
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