gitlab-org--gitlab-foss/spec/models/ci/build_dependencies_spec.rb

388 lines
13 KiB
Ruby

# frozen_string_literal: true
require 'spec_helper'
RSpec.describe Ci::BuildDependencies do
let_it_be(:user) { create(:user) }
let_it_be(:project, reload: true) { create(:project, :repository) }
let_it_be(:pipeline, reload: true) do
create(:ci_pipeline, project: project,
sha: project.commit.id,
ref: project.default_branch,
status: 'success')
end
let!(:build) { create(:ci_build, pipeline: pipeline, name: 'build', stage_idx: 0, stage: 'build') }
let!(:rspec_test) { create(:ci_build, :success, pipeline: pipeline, name: 'rspec', stage_idx: 1, stage: 'test') }
let!(:rubocop_test) { create(:ci_build, pipeline: pipeline, name: 'rubocop', stage_idx: 1, stage: 'test') }
let!(:staging) { create(:ci_build, pipeline: pipeline, name: 'staging', stage_idx: 2, stage: 'deploy') }
context 'for local dependencies' do
subject { described_class.new(job).all }
describe 'jobs from previous stages' do
context 'when job is in the first stage' do
let(:job) { build }
it { is_expected.to be_empty }
end
context 'when job is in the second stage' do
let(:job) { rspec_test }
it 'contains all jobs from the first stage' do
is_expected.to contain_exactly(build)
end
end
context 'when job is in the last stage' do
let(:job) { staging }
it 'contains all jobs from all previous stages' do
is_expected.to contain_exactly(build, rspec_test, rubocop_test)
end
context 'when a job is retried' do
before do
project.add_developer(user)
end
let!(:retried_job) { Ci::RetryJobService.new(rspec_test.project, user).execute(rspec_test)[:job] }
it 'contains the retried job instead of the original one' do
is_expected.to contain_exactly(build, retried_job, rubocop_test)
end
end
end
context 'when needs refer to jobs from the same stage' do
let(:job) do
create(:ci_build,
pipeline: pipeline,
name: 'dag_job',
scheduling_type: :dag,
stage_idx: 2,
stage: 'deploy'
)
end
before do
create(:ci_build_need, build: job, name: 'staging', artifacts: true)
end
it { is_expected.to contain_exactly(staging) }
end
end
describe 'jobs from specified dependencies' do
let(:dependencies) { }
let(:needs) { }
let!(:job) do
scheduling_type = needs.present? ? :dag : :stage
create(:ci_build,
pipeline: pipeline,
name: 'final',
scheduling_type: scheduling_type,
stage_idx: 3,
stage: 'deploy',
options: { dependencies: dependencies }
)
end
before do
needs.to_a.each do |need|
create(:ci_build_need, build: job, **need)
end
end
context 'when dependencies are defined' do
let(:dependencies) { %w(rspec staging) }
it { is_expected.to contain_exactly(rspec_test, staging) }
end
context 'when needs are defined' do
let(:needs) do
[
{ name: 'build', artifacts: true },
{ name: 'rspec', artifacts: true },
{ name: 'staging', artifacts: true }
]
end
it { is_expected.to contain_exactly(build, rspec_test, staging) }
end
context 'when need artifacts are defined' do
let(:needs) do
[
{ name: 'build', artifacts: true },
{ name: 'rspec', artifacts: false },
{ name: 'staging', artifacts: true }
]
end
it { is_expected.to contain_exactly(build, staging) }
end
context 'when needs and dependencies are defined' do
let(:dependencies) { %w(rspec staging) }
let(:needs) do
[
{ name: 'build', artifacts: true },
{ name: 'rspec', artifacts: true },
{ name: 'staging', artifacts: true }
]
end
it { is_expected.to contain_exactly(rspec_test, staging) }
end
context 'when needs and dependencies contradict' do
let(:dependencies) { %w(rspec staging) }
let(:needs) do
[
{ name: 'build', artifacts: true },
{ name: 'rspec', artifacts: false },
{ name: 'staging', artifacts: true }
]
end
it 'returns only the intersection' do
is_expected.to contain_exactly(staging)
end
end
context 'when nor dependencies or needs are defined' do
it 'returns the jobs from previous stages' do
is_expected.to contain_exactly(build, rspec_test, rubocop_test, staging)
end
end
end
end
context 'for cross_pipeline dependencies' do
let!(:job) do
create(:ci_build,
pipeline: pipeline,
name: 'build_with_pipeline_dependency',
options: { cross_dependencies: dependencies })
end
subject { described_class.new(job) }
let(:cross_pipeline_deps) { subject.all }
context 'when dependency specifications are valid' do
context 'when pipeline exists in the hierarchy' do
let!(:pipeline) { create(:ci_pipeline, child_of: parent_pipeline) }
let!(:parent_pipeline) { create(:ci_pipeline, project: project) }
context 'when job exists' do
let(:dependencies) do
[{ pipeline: parent_pipeline.id.to_s, job: upstream_job.name, artifacts: true }]
end
let!(:upstream_job) { create(:ci_build, :success, pipeline: parent_pipeline) }
it { expect(cross_pipeline_deps).to contain_exactly(upstream_job) }
it { is_expected.to be_valid }
context 'when pipeline and job are specified via variables' do
let(:dependencies) do
[{ pipeline: '$parent_pipeline_ID', job: '$UPSTREAM_JOB', artifacts: true }]
end
before do
job.yaml_variables.push(key: 'parent_pipeline_ID', value: parent_pipeline.id.to_s, public: true)
job.yaml_variables.push(key: 'UPSTREAM_JOB', value: upstream_job.name, public: true)
job.save!
end
it { expect(cross_pipeline_deps).to contain_exactly(upstream_job) }
it { is_expected.to be_valid }
end
end
context 'when same job names exist in other pipelines in the hierarchy' do
let(:cross_pipeline_limit) do
::Gitlab::Ci::Config::Entry::Needs::NEEDS_CROSS_PIPELINE_DEPENDENCIES_LIMIT
end
let(:sibling_pipeline) { create(:ci_pipeline, child_of: parent_pipeline) }
before do
cross_pipeline_limit.times do |index|
create(:ci_build, :success,
pipeline: parent_pipeline, name: "dependency-#{index}",
stage_idx: 1, stage: 'build', user: user
)
create(:ci_build, :success,
pipeline: sibling_pipeline, name: "dependency-#{index}",
stage_idx: 1, stage: 'build', user: user
)
end
end
let(:dependencies) do
[
{ pipeline: parent_pipeline.id.to_s, job: 'dependency-0', artifacts: true },
{ pipeline: parent_pipeline.id.to_s, job: 'dependency-1', artifacts: true },
{ pipeline: parent_pipeline.id.to_s, job: 'dependency-2', artifacts: true },
{ pipeline: sibling_pipeline.id.to_s, job: 'dependency-3', artifacts: true },
{ pipeline: sibling_pipeline.id.to_s, job: 'dependency-4', artifacts: true },
{ pipeline: sibling_pipeline.id.to_s, job: 'dependency-5', artifacts: true }
]
end
it 'returns a limited number of dependencies with the right match' do
expect(job.options[:cross_dependencies].size).to eq(cross_pipeline_limit.next)
expect(cross_pipeline_deps.size).to eq(cross_pipeline_limit)
expect(cross_pipeline_deps.map { |dep| [dep.pipeline_id, dep.name] }).to contain_exactly(
[parent_pipeline.id, 'dependency-0'],
[parent_pipeline.id, 'dependency-1'],
[parent_pipeline.id, 'dependency-2'],
[sibling_pipeline.id, 'dependency-3'],
[sibling_pipeline.id, 'dependency-4'])
end
end
context 'when job does not exist' do
let(:dependencies) do
[{ pipeline: parent_pipeline.id.to_s, job: 'non-existent', artifacts: true }]
end
it { expect(cross_pipeline_deps).to be_empty }
it { is_expected.not_to be_valid }
end
end
context 'when pipeline does not exist' do
let(:dependencies) do
[{ pipeline: '123', job: 'non-existent', artifacts: true }]
end
it { expect(cross_pipeline_deps).to be_empty }
it { is_expected.not_to be_valid }
end
context 'when jobs exist in different pipelines in the hierarchy' do
let!(:pipeline) { create(:ci_pipeline, child_of: parent_pipeline) }
let!(:parent_pipeline) { create(:ci_pipeline, project: project) }
let!(:parent_job) { create(:ci_build, :success, name: 'parent_job', pipeline: parent_pipeline) }
let!(:sibling_pipeline) { create(:ci_pipeline, child_of: parent_pipeline) }
let!(:sibling_job) { create(:ci_build, :success, name: 'sibling_job', pipeline: sibling_pipeline) }
context 'when pipeline and jobs dependencies are mismatched' do
let(:dependencies) do
[
{ pipeline: parent_pipeline.id.to_s, job: sibling_job.name, artifacts: true },
{ pipeline: sibling_pipeline.id.to_s, job: parent_job.name, artifacts: true }
]
end
it { expect(cross_pipeline_deps).to be_empty }
it { is_expected.not_to be_valid }
context 'when dependencies contain a valid pair' do
let(:dependencies) do
[
{ pipeline: parent_pipeline.id.to_s, job: sibling_job.name, artifacts: true },
{ pipeline: sibling_pipeline.id.to_s, job: parent_job.name, artifacts: true },
{ pipeline: sibling_pipeline.id.to_s, job: sibling_job.name, artifacts: true }
]
end
it 'filters out the invalid ones' do
expect(cross_pipeline_deps).to contain_exactly(sibling_job)
end
it { is_expected.not_to be_valid }
end
end
end
context 'when job and pipeline exist outside the hierarchy' do
let!(:pipeline) { create(:ci_pipeline, project: project) }
let!(:another_pipeline) { create(:ci_pipeline, project: project) }
let!(:dependency) { create(:ci_build, :success, pipeline: another_pipeline) }
let(:dependencies) do
[{ pipeline: another_pipeline.id.to_s, job: dependency.name, artifacts: true }]
end
it 'ignores jobs outside the pipeline hierarchy' do
expect(cross_pipeline_deps).to be_empty
end
it { is_expected.not_to be_valid }
end
context 'when current pipeline is specified' do
let!(:pipeline) { create(:ci_pipeline, project: project) }
let!(:dependency) { create(:ci_build, :success, pipeline: pipeline) }
let(:dependencies) do
[{ pipeline: pipeline.id.to_s, job: dependency.name, artifacts: true }]
end
it 'ignores jobs from the current pipeline as simple needs should be used instead' do
expect(cross_pipeline_deps).to be_empty
end
it { is_expected.not_to be_valid }
end
end
context 'when artifacts:false' do
let!(:pipeline) { create(:ci_pipeline, child_of: parent_pipeline) }
let!(:parent_pipeline) { create(:ci_pipeline, project: project) }
let!(:parent_job) { create(:ci_build, :success, name: 'parent_job', pipeline: parent_pipeline) }
let(:dependencies) do
[{ pipeline: parent_pipeline.id.to_s, job: parent_job.name, artifacts: false }]
end
it { expect(cross_pipeline_deps).to be_empty }
it { is_expected.to be_valid } # we simply ignore it
end
end
describe '#all' do
let!(:job) do
create(:ci_build, pipeline: pipeline, name: 'deploy', stage_idx: 3, stage: 'deploy')
end
let(:dependencies) { described_class.new(job) }
subject { dependencies.all }
it 'returns the union of all local dependencies and any cross project dependencies' do
expect(dependencies).to receive(:local).and_return([1, 2, 3])
expect(dependencies).to receive(:cross_project).and_return([3, 4])
expect(subject).to contain_exactly(1, 2, 3, 4)
end
end
describe '#valid?' do
subject { described_class.new(job).valid? }
let(:job) { rspec_test }
it { is_expected.to eq(true) }
context 'when a local dependency is invalid' do
before do
build.update_column(:erased_at, Time.current)
end
it { is_expected.to eq(false) }
end
end
end