debian-mirror-gitlab/lib/gitlab/config/entry/validators.rb

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

441 lines
14 KiB
Ruby
Raw Normal View History

2019-02-15 15:39:39 +05:30
# frozen_string_literal: true
module Gitlab
module Config
module Entry
module Validators
class AllowedKeysValidator < ActiveModel::EachValidator
def validate_each(record, attribute, value)
unknown_keys = value.try(:keys).to_a - options[:in]
if unknown_keys.any?
record.errors.add(attribute, "contains unknown keys: " +
unknown_keys.join(', '))
end
end
end
2019-07-07 11:18:12 +05:30
class DisallowedKeysValidator < ActiveModel::EachValidator
def validate_each(record, attribute, value)
present_keys = value.try(:keys).to_a & options[:in]
if present_keys.any?
2019-12-04 20:38:33 +05:30
message = options[:message] || "contains disallowed keys"
message += ": #{present_keys.join(', ')}"
record.errors.add(attribute, message)
2019-10-12 21:52:04 +05:30
end
end
end
class RequiredKeysValidator < ActiveModel::EachValidator
def validate_each(record, attribute, value)
present_keys = options[:in] - value.try(:keys).to_a
if present_keys.any?
record.errors.add(attribute, "missing required keys: " +
present_keys.join(', '))
2019-07-07 11:18:12 +05:30
end
end
end
2022-05-07 20:08:51 +05:30
class MutuallyExclusiveKeysValidator < ActiveModel::EachValidator
def validate_each(record, attribute, value)
mutually_exclusive_keys = value.try(:keys).to_a & options[:in]
if mutually_exclusive_keys.length > 1
record.errors.add(attribute, "please use only one the following keys: " +
mutually_exclusive_keys.join(', '))
end
end
end
2019-02-15 15:39:39 +05:30
class AllowedValuesValidator < ActiveModel::EachValidator
def validate_each(record, attribute, value)
unless options[:in].include?(value.to_s)
record.errors.add(attribute, "unknown value: #{value}")
end
end
end
class AllowedArrayValuesValidator < ActiveModel::EachValidator
def validate_each(record, attribute, value)
2019-09-04 21:01:54 +05:30
unknown_values = value - options[:in]
unless unknown_values.empty?
2019-02-15 15:39:39 +05:30
record.errors.add(attribute, "contains unknown values: " +
2019-09-04 21:01:54 +05:30
unknown_values.join(', '))
2019-02-15 15:39:39 +05:30
end
end
end
class ArrayOfStringsValidator < ActiveModel::EachValidator
include LegacyValidationHelpers
def validate_each(record, attribute, value)
2019-12-26 22:10:19 +05:30
valid = validate_array_of_strings(value)
record.errors.add(attribute, 'should be an array of strings') unless valid
if valid && options[:with]
unless value.all? { |v| v =~ options[:with] }
message = options[:message] || 'contains elements that do not match the format'
record.errors.add(attribute, message)
end
2019-02-15 15:39:39 +05:30
end
end
end
2019-12-04 20:38:33 +05:30
class ArrayOfHashesValidator < ActiveModel::EachValidator
include LegacyValidationHelpers
def validate_each(record, attribute, value)
2021-10-27 15:23:28 +05:30
unless validate_array_of_hashes(value)
2019-12-04 20:38:33 +05:30
record.errors.add(attribute, 'should be an array of hashes')
end
end
2021-10-27 15:23:28 +05:30
private
def validate_array_of_hashes(value)
value.is_a?(Array) && value.all? { |obj| obj.is_a?(Hash) }
end
end
2021-11-11 11:23:49 +05:30
class NestedArrayOfHashesOrArraysValidator < ArrayOfHashesValidator
2021-10-27 15:23:28 +05:30
include NestedArrayHelpers
def validate_each(record, attribute, value)
2021-11-11 11:23:49 +05:30
max_level = options.fetch(:max_level, 1)
unless validate_nested_array(value, max_level, &method(:validate_hash))
2021-10-27 15:23:28 +05:30
record.errors.add(attribute, 'should be an array containing hashes and arrays of hashes')
end
end
2021-11-11 11:23:49 +05:30
private
def validate_hash(value)
value.is_a?(Hash)
end
2019-12-04 20:38:33 +05:30
end
2019-07-07 11:18:12 +05:30
class ArrayOrStringValidator < ActiveModel::EachValidator
def validate_each(record, attribute, value)
unless value.is_a?(Array) || value.is_a?(String)
record.errors.add(attribute, 'should be an array or a string')
end
end
end
2019-02-15 15:39:39 +05:30
class BooleanValidator < ActiveModel::EachValidator
include LegacyValidationHelpers
def validate_each(record, attribute, value)
unless validate_boolean(value)
record.errors.add(attribute, 'should be a boolean value')
end
end
end
class DurationValidator < ActiveModel::EachValidator
include LegacyValidationHelpers
def validate_each(record, attribute, value)
2020-10-24 23:57:45 +05:30
unless validate_duration(value, options[:parser])
2019-02-15 15:39:39 +05:30
record.errors.add(attribute, 'should be a duration')
end
if options[:limit]
2020-10-24 23:57:45 +05:30
unless validate_duration_limit(value, options[:limit], options[:parser])
2019-02-15 15:39:39 +05:30
record.errors.add(attribute, 'should not exceed the limit')
end
end
end
end
class HashOrStringValidator < ActiveModel::EachValidator
def validate_each(record, attribute, value)
unless value.is_a?(Hash) || value.is_a?(String)
record.errors.add(attribute, 'should be a hash or a string')
end
end
end
class HashOrIntegerValidator < ActiveModel::EachValidator
def validate_each(record, attribute, value)
unless value.is_a?(Hash) || value.is_a?(Integer)
record.errors.add(attribute, 'should be a hash or an integer')
end
end
end
2021-02-22 17:27:13 +05:30
class HashOrBooleanValidator < ActiveModel::EachValidator
include LegacyValidationHelpers
def validate_each(record, attribute, value)
unless value.is_a?(Hash) || validate_boolean(value)
record.errors.add(attribute, 'should be a hash or a boolean value')
end
end
end
2019-02-15 15:39:39 +05:30
class KeyValidator < ActiveModel::EachValidator
include LegacyValidationHelpers
def validate_each(record, attribute, value)
if validate_string(value)
validate_path(record, attribute, value)
else
record.errors.add(attribute, 'should be a string or symbol')
end
end
private
def validate_path(record, attribute, value)
path = CGI.unescape(value.to_s)
if path.include?('/')
record.errors.add(attribute, 'cannot contain the "/" character')
elsif path == '.' || path == '..'
record.errors.add(attribute, 'cannot be "." or ".."')
end
end
end
2021-02-22 17:27:13 +05:30
class ArrayOfIntegersOrIntegerValidator < ActiveModel::EachValidator
include LegacyValidationHelpers
def validate_each(record, attribute, value)
unless validate_integer(value) || validate_array_of_integers(value)
record.errors.add(attribute, 'should be an array of integers or an integer')
end
end
private
def validate_array_of_integers(values)
values.is_a?(Array) && values.all? { |value| validate_integer(value) }
end
end
2019-02-15 15:39:39 +05:30
class RegexpValidator < ActiveModel::EachValidator
include LegacyValidationHelpers
def validate_each(record, attribute, value)
unless validate_regexp(value)
2022-04-04 11:22:00 +05:30
record.errors.add(attribute, 'must be a regular expression with re2 syntax')
2019-02-15 15:39:39 +05:30
end
end
private
2019-04-03 18:18:56 +05:30
def matches_syntax?(value)
Gitlab::UntrustedRegexp::RubySyntax.matches_syntax?(value)
2019-02-15 15:39:39 +05:30
end
def validate_regexp(value)
2019-04-03 18:18:56 +05:30
matches_syntax?(value) &&
2022-05-07 20:08:51 +05:30
Gitlab::UntrustedRegexp::RubySyntax.valid?(value)
2019-02-15 15:39:39 +05:30
end
end
class ArrayOfStringsOrRegexpsValidator < RegexpValidator
def validate_each(record, attribute, value)
unless validate_array_of_strings_or_regexps(value)
2022-04-04 11:22:00 +05:30
record.errors.add(attribute, validation_message)
2019-02-15 15:39:39 +05:30
end
end
private
2022-04-04 11:22:00 +05:30
def validation_message
'should be an array of strings or regular expressions using re2 syntax'
end
2019-02-15 15:39:39 +05:30
def validate_array_of_strings_or_regexps(values)
values.is_a?(Array) && values.all?(&method(:validate_string_or_regexp))
end
def validate_string_or_regexp(value)
return false unless value.is_a?(String)
2019-04-03 18:18:56 +05:30
return validate_regexp(value) if matches_syntax?(value)
2019-02-15 15:39:39 +05:30
true
end
end
class ArrayOfStringsOrStringValidator < RegexpValidator
def validate_each(record, attribute, value)
unless validate_array_of_strings_or_string(value)
record.errors.add(attribute, 'should be an array of strings or a string')
end
end
private
def validate_array_of_strings_or_string(values)
validate_array_of_strings(values) || validate_string(values)
end
end
2021-03-11 19:13:27 +05:30
class StringOrNestedArrayOfStringsValidator < ActiveModel::EachValidator
include LegacyValidationHelpers
include NestedArrayHelpers
2019-12-26 22:10:19 +05:30
2021-03-11 19:13:27 +05:30
def validate_each(record, attribute, value)
max_level = options.fetch(:max_level, 1)
2019-12-26 22:10:19 +05:30
2021-03-11 19:13:27 +05:30
unless validate_string(value) || validate_nested_array(value, max_level, &method(:validate_string))
record.errors.add(attribute, "should be a string or a nested array of strings up to #{max_level} levels deep")
end
2019-12-26 22:10:19 +05:30
end
end
2019-02-15 15:39:39 +05:30
class TypeValidator < ActiveModel::EachValidator
def validate_each(record, attribute, value)
type = options[:with]
raise unless type.is_a?(Class)
unless value.is_a?(type)
message = options[:message] || "should be a #{type.name}"
record.errors.add(attribute, message)
end
end
end
class VariablesValidator < ActiveModel::EachValidator
include LegacyValidationHelpers
def validate_each(record, attribute, value)
2020-10-24 23:57:45 +05:30
if options[:array_values]
validate_key_array_values(record, attribute, value)
2021-01-03 14:25:43 +05:30
elsif options[:allowed_value_data]
validate_key_hash_values(record, attribute, value, options[:allowed_value_data])
2020-10-24 23:57:45 +05:30
else
validate_key_values(record, attribute, value)
end
end
def validate_key_values(record, attribute, value)
2019-02-15 15:39:39 +05:30
unless validate_variables(value)
record.errors.add(attribute, 'should be a hash of key value pairs')
end
end
2020-10-24 23:57:45 +05:30
def validate_key_array_values(record, attribute, value)
unless validate_array_value_variables(value)
record.errors.add(attribute, 'should be a hash of key value pairs, value can be an array')
end
end
2021-01-03 14:25:43 +05:30
def validate_key_hash_values(record, attribute, value, allowed_value_data)
unless validate_string_or_hash_value_variables(value, allowed_value_data)
record.errors.add(attribute, 'should be a hash of key value pairs, value can be a hash')
end
end
2019-02-15 15:39:39 +05:30
end
2019-07-07 11:18:12 +05:30
2019-12-04 20:38:33 +05:30
class ExpressionValidator < ActiveModel::EachValidator
def validate_each(record, attribute, value)
unless value.is_a?(String) && ::Gitlab::Ci::Pipeline::Expression::Statement.new(value).valid?
record.errors.add(attribute, 'Invalid expression syntax')
end
end
end
2019-07-07 11:18:12 +05:30
class PortNamePresentAndUniqueValidator < ActiveModel::EachValidator
def validate_each(record, attribute, value)
return unless value.is_a?(Array)
ports_size = value.count
return if ports_size <= 1
named_ports = value.select { |e| e.is_a?(Hash) }.map { |e| e[:name] }.compact.map(&:downcase)
if ports_size != named_ports.size
record.errors.add(attribute, 'when there is more than one port, a unique name should be added')
end
if ports_size != named_ports.uniq.size
record.errors.add(attribute, 'each port name must be different')
end
end
end
class PortUniqueValidator < ActiveModel::EachValidator
def validate_each(record, attribute, value)
value = ports(value)
return unless value.is_a?(Array)
ports_size = value.count
return if ports_size <= 1
if transform_ports(value).size != ports_size
record.errors.add(attribute, 'each port number can only be referenced once')
end
end
private
def ports(current_data)
current_data
end
def transform_ports(raw_ports)
raw_ports.map do |port|
case port
when Integer
port
when Hash
port[:number]
end
end.uniq
end
end
class JobPortUniqueValidator < PortUniqueValidator
private
def ports(current_data)
return unless current_data.is_a?(Hash)
(image_ports(current_data) + services_ports(current_data)).compact
end
def image_ports(current_data)
return [] unless current_data[:image].is_a?(Hash)
current_data.dig(:image, :ports).to_a
end
def services_ports(current_data)
current_data.dig(:services).to_a.flat_map { |service| service.is_a?(Hash) ? service[:ports] : nil }
end
end
class ServicesWithPortsAliasUniqueValidator < ActiveModel::EachValidator
def validate_each(record, attribute, value)
current_aliases = aliases(value)
return if current_aliases.empty?
unless aliases_unique?(current_aliases)
record.errors.add(:config, 'alias must be unique in services with ports')
end
end
private
def aliases(value)
value.select { |s| s.is_a?(Hash) && s[:ports] }.pluck(:alias) # rubocop:disable CodeReuse/ActiveRecord
end
def aliases_unique?(aliases)
aliases.size == aliases.uniq.size
end
end
2019-02-15 15:39:39 +05:30
end
end
end
end