Commit e37777e4 authored by Matija Čupić's avatar Matija Čupić

Validate need types in config

Validates need types in Job and Bridge config.

- Only Pipeline type needs are allowed in regular jobs
- Only Bridge and Pipeline type needs are allowed in bridge jobs
- Only one Bridge type need is allowed in a bridge job
parent da2e373a
...@@ -29,6 +29,22 @@ module EE ...@@ -29,6 +29,22 @@ module EE
end end
end end
validate do
next unless needs.present?
if [needs].flatten.any? { |need| !(Entry::Need::Bridge.matching?(need) || ::Gitlab::Ci::Config::Entry::Need::Pipeline.matching?(need)) }
errors.add(:needs, 'can only have bridge or pipeline type needs')
end
end
validate do
next unless needs.present?
if [needs].flatten.count { |need| Entry::Need::Bridge.matching?(need) } > 1
errors.add(:needs, 'can only have one bridge type needs')
end
end
with_options allow_nil: true do with_options allow_nil: true do
validates :when, validates :when,
inclusion: { in: %w[on_success on_failure always], inclusion: { in: %w[on_success on_failure always],
......
...@@ -25,17 +25,12 @@ module EE ...@@ -25,17 +25,12 @@ module EE
validates :pipeline, type: String, presence: true validates :pipeline, type: String, presence: true
end end
def type def self.matching?(config)
:bridge config.is_a?(Hash)
end end
end
module UnknownStrategy def type
extend ::Gitlab::Utils::Override :bridge
override :errors
def errors
["#{location} has to be a string, symbol or hash"]
end end
end end
end end
......
# frozen_string_literal: true
module EE
module Gitlab
module Ci
module Config
module Entry
module Needs
extend ActiveSupport::Concern
class_methods do
extend ::Gitlab::Utils::Override
override :all_types
def all_types
super + [Entry::Need::Bridge]
end
end
end
end
end
end
end
end
...@@ -160,6 +160,38 @@ describe EE::Gitlab::Ci::Config::Entry::Bridge do ...@@ -160,6 +160,38 @@ describe EE::Gitlab::Ci::Config::Entry::Bridge do
end end
end end
context 'when bridge has bridge and pipeline needs' do
let(:config) do
{
trigger: 'other-project',
needs: ['some_job', { pipeline: 'some/other_project' }]
}
end
describe '#valid?' do
it { is_expected.to be_valid }
end
end
context 'when bridge has more than one valid bridge needs' do
let(:config) do
{
trigger: 'other-project',
needs: [{ pipeline: 'some/project' }, { pipeline: 'some/other_project' }]
}
end
describe '#valid?' do
it { is_expected.not_to be_valid }
end
describe '#errors' do
it 'returns an error about too many bridge needs' do
expect(subject.errors).to contain_exactly('bridge needs can only have one bridge type needs')
end
end
end
context 'when bridge config contains unknown keys' do context 'when bridge config contains unknown keys' do
let(:config) { { unknown: 123 } } let(:config) { { unknown: 123 } }
......
...@@ -28,23 +28,8 @@ describe ::Gitlab::Ci::Config::Entry::Need do ...@@ -28,23 +28,8 @@ describe ::Gitlab::Ci::Config::Entry::Need do
describe '#errors' do describe '#errors' do
it 'is returns an error about an empty config' do it 'is returns an error about an empty config' do
expect(subject.errors.first) expect(subject.errors)
.to end_with("bridge config can't be blank") .to include("bridge config can't be blank")
end
end
end
context 'when need is the wrong type' do
let(:config) { 123 }
describe '#valid?' do
it { is_expected.not_to be_valid }
end
describe '#errors' do
it 'is returns an error about an empty config' do
expect(subject.errors.first)
.to end_with('has to be a string, symbol or hash')
end end
end end
end end
......
...@@ -27,10 +27,8 @@ describe ::Gitlab::Ci::Config::Entry::Needs do ...@@ -27,10 +27,8 @@ describe ::Gitlab::Ci::Config::Entry::Needs do
describe '#errors' do describe '#errors' do
it 'returns error about incorrect type' do it 'returns error about incorrect type' do
error_message = Gitlab.ee? ? 'need has to be a string, symbol or hash' : 'need has to be a string or symbol'
expect(needs.errors) expect(needs.errors)
.to include error_message .to contain_exactly('needs need has an unsupported type')
end end
end end
end end
...@@ -54,7 +52,7 @@ describe ::Gitlab::Ci::Config::Entry::Needs do ...@@ -54,7 +52,7 @@ describe ::Gitlab::Ci::Config::Entry::Needs do
describe '#value' do describe '#value' do
it 'returns key value' do it 'returns key value' do
expect(needs.value).to eq(pipeline: ['first_job_name'], bridge: { pipeline: 'some/project' }) expect(needs.value).to eq(pipeline: [{ name: 'first_job_name' }], bridge: { pipeline: 'some/project' })
end end
end end
......
...@@ -57,6 +57,14 @@ module Gitlab ...@@ -57,6 +57,14 @@ module Gitlab
validates :start_in, duration: { limit: '1 day' }, if: :delayed? validates :start_in, duration: { limit: '1 day' }, if: :delayed?
validates :start_in, absence: true, if: -> { has_rules? || !delayed? } validates :start_in, absence: true, if: -> { has_rules? || !delayed? }
validate do
next unless needs.present?
if [needs].flatten.any? { |need| !Entry::Need::Pipeline.matching?(need) }
errors.add(:needs, 'can only have pipeline type needs')
end
end
validate do validate do
next unless dependencies.present? next unless dependencies.present?
next unless needs.present? next unless needs.present?
......
...@@ -14,6 +14,10 @@ module Gitlab ...@@ -14,6 +14,10 @@ module Gitlab
validates :config, presence: true validates :config, presence: true
end end
def self.matching?(config)
config.is_a?(String) || config.is_a?(Symbol)
end
def type def type
:pipeline :pipeline
end end
...@@ -24,9 +28,6 @@ module Gitlab ...@@ -24,9 +28,6 @@ module Gitlab
end end
class UnknownStrategy < ::Gitlab::Config::Entry::Node class UnknownStrategy < ::Gitlab::Config::Entry::Node
def errors
["#{location} has to be a string or symbol"]
end
end end
end end
end end
...@@ -34,5 +35,4 @@ module Gitlab ...@@ -34,5 +35,4 @@ module Gitlab
end end
end end
::Gitlab::Ci::Config::Entry::Need.prepend_if_ee('::EE::Gitlab::Ci::Config::Entry::Need') # rubocop: disable Cop/InjectEnterpriseEditionModule ::Gitlab::Ci::Config::Entry::Need.prepend_if_ee('::EE::Gitlab::Ci::Config::Entry::Need')
::Gitlab::Ci::Config::Entry::Need::UnknownStrategy.prepend_if_ee('::EE::Gitlab::Ci::Config::Entry::Need::UnknownStrategy')
...@@ -12,6 +12,28 @@ module Gitlab ...@@ -12,6 +12,28 @@ module Gitlab
validations do validations do
validates :config, presence: true validates :config, presence: true
validate do
[config].flatten.each do |need|
if Needs.find_type(need).nil?
errors.add(:need, 'has an unsupported type')
end
end
end
end
TYPES = [Entry::Need::Pipeline].freeze
private_constant :TYPES
def self.all_types
TYPES
end
def self.find_type(config)
self.all_types.find do |type|
type.matching?(config)
end
end end
def compose!(deps = nil) def compose!(deps = nil)
...@@ -30,12 +52,18 @@ module Gitlab ...@@ -30,12 +52,18 @@ module Gitlab
end end
def value def value
@entries.values.group_by(&:type).transform_values do |values| values = @entries.values.group_by(&:type).transform_values do |values|
values.map(&:value) values.map(&:value)
end end
values.tap do |values_hash|
values_hash[:bridge] = values_hash[:bridge].first if values_hash[:bridge]
end
end end
end end
end end
end end
end end
end end
::Gitlab::Ci::Config::Entry::Needs.prepend_if_ee('::EE::Gitlab::Ci::Config::Entry::Needs')
...@@ -164,12 +164,14 @@ module Gitlab ...@@ -164,12 +164,14 @@ module Gitlab
stage_index = @stages.index(job[:stage]) stage_index = @stages.index(job[:stage])
job[:needs][:pipeline].each do |need| job[:needs][:pipeline].each do |need|
raise ValidationError, "#{name} job: undefined need: #{need}" unless @jobs[need.to_sym] need_job_name = need[:name]
needs_stage_index = @stages.index(@jobs[need.to_sym][:stage]) raise ValidationError, "#{name} job: undefined need: #{need_job_name}" unless @jobs[need_job_name.to_sym]
needs_stage_index = @stages.index(@jobs[need_job_name.to_sym][:stage])
unless needs_stage_index.present? && needs_stage_index < stage_index unless needs_stage_index.present? && needs_stage_index < stage_index
raise ValidationError, "#{name} job: need #{need} is not defined in prior stages" raise ValidationError, "#{name} job: need #{need_job_name} is not defined in prior stages"
end end
end end
end end
......
...@@ -413,6 +413,21 @@ describe Gitlab::Ci::Config::Entry::Job do ...@@ -413,6 +413,21 @@ describe Gitlab::Ci::Config::Entry::Job do
expect(entry.errors).to include 'job config missing required keys: stage' expect(entry.errors).to include 'job config missing required keys: stage'
end end
end end
context 'when needs is bridge type' do
let(:config) do
{
script: 'echo',
stage: 'test',
needs: { pipeline: 'some/project' }
}
end
it 'returns error about invalid needs type' do
expect(entry).not_to be_valid
expect(entry.errors).to contain_exactly('job needs can only have pipeline type needs')
end
end
end end
context 'when timeout value is not correct' do context 'when timeout value is not correct' do
......
...@@ -14,7 +14,7 @@ describe ::Gitlab::Ci::Config::Entry::Need do ...@@ -14,7 +14,7 @@ describe ::Gitlab::Ci::Config::Entry::Need do
describe '#value' do describe '#value' do
it 'returns job needs configuration' do it 'returns job needs configuration' do
expect(need.value).to eq('job_name') expect(need.value).to eq(name: 'job_name')
end end
end end
end end
...@@ -28,7 +28,7 @@ describe ::Gitlab::Ci::Config::Entry::Need do ...@@ -28,7 +28,7 @@ describe ::Gitlab::Ci::Config::Entry::Need do
describe '#value' do describe '#value' do
it 'returns job needs configuration' do it 'returns job needs configuration' do
expect(need.value).to eq('job_name') expect(need.value).to eq(name: :job_name)
end end
end end
end end
...@@ -42,25 +42,8 @@ describe ::Gitlab::Ci::Config::Entry::Need do ...@@ -42,25 +42,8 @@ describe ::Gitlab::Ci::Config::Entry::Need do
describe '#errors' do describe '#errors' do
it 'is returns an error about an empty config' do it 'is returns an error about an empty config' do
expect(need.errors.first) expect(need.errors)
.to end_with("pipeline config can't be blank") .to contain_exactly("pipeline config can't be blank")
end
end
end
context 'when need is not a string' do
let(:config) { 123 }
describe '#valid?' do
it { is_expected.not_to be_valid }
end
describe '#errors' do
it 'is returns an error about an empty config' do
error_message = Gitlab.ee? ? 'has to be a string, symbol or hash' : 'has to be a string or symbol'
expect(need.errors.first)
.to end_with(error_message)
end end
end end
end end
......
...@@ -27,10 +27,8 @@ describe ::Gitlab::Ci::Config::Entry::Needs do ...@@ -27,10 +27,8 @@ describe ::Gitlab::Ci::Config::Entry::Needs do
describe '#errors' do describe '#errors' do
it 'returns error about incorrect type' do it 'returns error about incorrect type' do
error_message = Gitlab.ee? ? 'need has to be a string, symbol or hash' : 'need has to be a string or symbol'
expect(needs.errors) expect(needs.errors)
.to include error_message .to contain_exactly('needs need has an unsupported type')
end end
end end
end end
...@@ -46,7 +44,7 @@ describe ::Gitlab::Ci::Config::Entry::Needs do ...@@ -46,7 +44,7 @@ describe ::Gitlab::Ci::Config::Entry::Needs do
describe '#value' do describe '#value' do
it 'returns key value' do it 'returns key value' do
expect(needs.value).to eq(pipeline: %w[first_job_name second_job_name]) expect(needs.value).to eq(pipeline: [{ name: 'first_job_name' }, { name: 'second_job_name' }])
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