# frozen_string_literal: true require 'fast_spec_helper' RSpec.describe Gitlab::Ci::Config::Normalizer do let(:job_name) { :rspec } let(:job_config) { { script: 'rspec', parallel: parallel_config, name: 'rspec', variables: variables_config } } let(:config) { { job_name => job_config } } describe '.normalize_jobs' do subject { described_class.new(config).normalize_jobs } shared_examples 'parallel dependencies' do context "when job has dependencies on parallelized jobs" do let(:config) do { job_name => job_config, other_job: { script: 'echo 1', dependencies: [job_name.to_s] } } end it "parallelizes dependencies" do expect(subject[:other_job][:dependencies]).to eq(expanded_job_names) end it "does not include original job name in #{context}" do expect(subject[:other_job][:dependencies]).not_to include(job_name) end end context "when there are dependencies which are both parallelized and not" do let(:config) do { job_name => job_config, other_job: { script: 'echo 1' }, final_job: { script: 'echo 1', dependencies: [job_name.to_s, "other_job"] } } end it "parallelizes dependencies" do expect(subject[:final_job][:dependencies]).to include(*expanded_job_names) end it "includes the regular job in dependencies" do expect(subject[:final_job][:dependencies]).to include('other_job') end end end shared_examples 'parallel needs' do let(:expanded_job_attributes) do expanded_job_names.map do |job_name| { name: job_name, extra: :key } end end context 'when job has needs on parallelized jobs' do let(:config) do { job_name => job_config, other_job: { script: 'echo 1', needs: { job: [ { name: job_name.to_s, extra: :key } ] } } } end it 'parallelizes needs' do expect(subject.dig(:other_job, :needs, :job)).to eq(expanded_job_attributes) end end context 'when there are dependencies which are both parallelized and not' do let(:config) do { job_name => job_config, other_job: { script: 'echo 1' }, final_job: { script: 'echo 1', needs: { job: [ { name: job_name.to_s, extra: :key }, { name: 'other_job', extra: :key } ] } } } end it 'parallelizes dependencies' do expect(subject.dig(:final_job, :needs, :job)).to include(*expanded_job_attributes) end it 'includes the regular job in dependencies' do expect(subject.dig(:final_job, :needs, :job)).to include(name: 'other_job', extra: :key) end end end context 'with parallel config as integer' do let(:variables_config) { {} } let(:parallel_config) { 5 } let(:expanded_job_names) do [ 'rspec 1/5', 'rspec 2/5', 'rspec 3/5', 'rspec 4/5', 'rspec 5/5' ] end it 'does not have original job' do is_expected.not_to include(job_name) end it 'has parallelized jobs' do is_expected.to include(*expanded_job_names.map(&:to_sym)) end it 'sets job instance in options' do expect(subject.values).to all(include(:instance)) end it 'parallelizes jobs with original config' do original_config = config[job_name] .except(:name) .deep_merge(parallel: { total: parallel_config }) configs = subject.values.map { |config| config.except(:name, :instance) } expect(configs).to all(eq(original_config)) end context 'when the job is not parallelized' do let(:job_config) { { script: 'rspec', name: 'rspec' } } it 'returns the same hash' do is_expected.to eq(config) end end context 'when there is a job with a slash in it' do let(:job_name) { :"rspec 35/2" } it 'properly parallelizes job names' do job_names = [ :"rspec 35/2 1/5", :"rspec 35/2 2/5", :"rspec 35/2 3/5", :"rspec 35/2 4/5", :"rspec 35/2 5/5" ] is_expected.to include(*job_names) end end it_behaves_like 'parallel dependencies' it_behaves_like 'parallel needs' end context 'with parallel matrix config' do let(:variables_config) do { USER_VARIABLE: 'user value' } end let(:parallel_config) do { matrix: [ { VAR_1: [1], VAR_2: [2, 3] } ] } end let(:expanded_job_names) do [ 'rspec 1/2', 'rspec 2/2' ] end it 'does not have original job' do is_expected.not_to include(job_name) end it 'has parallelized jobs' do is_expected.to include(*expanded_job_names.map(&:to_sym)) end it 'sets job instance in options' do expect(subject.values).to all(include(:instance)) end it 'sets job variables', :aggregate_failures do expect(subject.values[0]).to match( a_hash_including(variables: { VAR_1: 1, VAR_2: 2, USER_VARIABLE: 'user value' }) ) expect(subject.values[1]).to match( a_hash_including(variables: { VAR_1: 1, VAR_2: 3, USER_VARIABLE: 'user value' }) ) end it 'parallelizes jobs with original config' do configs = subject.values.map do |config| config.except(:name, :instance, :variables) end original_config = config[job_name] .except(:name, :variables) .deep_merge(parallel: { total: 2 }) expect(configs).to all(match(a_hash_including(original_config))) end it_behaves_like 'parallel dependencies' it_behaves_like 'parallel needs' end context 'when parallel config does not matches a factory' do let(:variables_config) { {} } let(:parallel_config) { } it 'does not alter the job config' do is_expected.to match(config) end end end end