gitlab_ci_yaml_processor.rb 11.6 KB
Newer Older
D
Douwe Maan 已提交
1 2
module Ci
  class GitlabCiYamlProcessor
3
    class ValidationError < StandardError; end
D
Douwe Maan 已提交
4

5
    include Gitlab::Ci::Config::Node::LegacyValidationHelpers
6

D
Douwe Maan 已提交
7 8
    DEFAULT_STAGES = %w(build test deploy)
    DEFAULT_STAGE = 'test'
9
    ALLOWED_YAML_KEYS = [:before_script, :after_script, :image, :services, :types, :stages, :variables, :cache]
10 11
    ALLOWED_JOB_KEYS = [:tags, :script, :only, :except, :type, :image, :services,
                        :allow_failure, :type, :stage, :when, :artifacts, :cache,
12 13
                        :dependencies, :before_script, :after_script, :variables,
                        :environment]
K
Kamil Trzcinski 已提交
14
    ALLOWED_CACHE_KEYS = [:key, :untracked, :paths]
15
    ALLOWED_ARTIFACTS_KEYS = [:name, :untracked, :paths, :when, :expire_in]
D
Douwe Maan 已提交
16

17
    attr_reader :after_script, :services, :path, :cache
D
Douwe Maan 已提交
18

19
    def initialize(config, path = nil)
20 21 22
      @ci_config = Gitlab::Ci::Config.new(config)
      @config = @ci_config.to_hash

23
      @path = path
D
Douwe Maan 已提交
24

25 26 27
      unless @ci_config.valid?
        raise ValidationError, @ci_config.errors.first
      end
D
Douwe Maan 已提交
28

29
      initial_parsing
D
Douwe Maan 已提交
30
      validate!
31
    rescue Gitlab::Ci::Config::Loader::FormatError => e
32
      raise ValidationError, e.message
D
Douwe Maan 已提交
33 34
    end

35
    def builds_for_stage_and_ref(stage, ref, tag = false, trigger_request = nil)
36 37 38 39
      builds.select do |build|
        build[:stage] == stage &&
          process?(build[:only], build[:except], ref, tag, trigger_request)
      end
D
Douwe Maan 已提交
40 41 42 43 44 45 46 47 48 49 50 51
    end

    def builds
      @jobs.map do |name, job|
        build_job(name, job)
      end
    end

    def stages
      @stages || DEFAULT_STAGES
    end

52
    def global_variables
53 54 55 56
      @variables
    end

    def job_variables(name)
G
Grzegorz Bizon 已提交
57 58 59
      job = @jobs[name.to_sym]
      return [] unless job

60
      job[:variables] || []
61 62
    end

D
Douwe Maan 已提交
63 64 65
    private

    def initial_parsing
66 67 68
      @before_script = @ci_config.before_script
      @image = @ci_config.image

69
      @after_script = @config[:after_script]
D
Douwe Maan 已提交
70 71 72 73
      @image = @config[:image]
      @services = @config[:services]
      @stages = @config[:stages] || @config[:types]
      @variables = @config[:variables] || {}
74
      @cache = @config[:cache]
75
      @jobs = {}
D
Douwe Maan 已提交
76

77
      @config.except!(*ALLOWED_YAML_KEYS)
T
Tomasz Maczukin 已提交
78
      @config.each { |name, param| add_job(name, param) }
D
Douwe Maan 已提交
79

80
      raise ValidationError, "Please define at least one job" if @jobs.none?
D
Douwe Maan 已提交
81 82
    end

83 84 85 86 87 88 89
    def add_job(name, job)
      return if name.to_s.start_with?('.')

      raise ValidationError, "Unknown parameter: #{name}" unless job.is_a?(Hash) && job.has_key?(:script)

      stage = job[:stage] || job[:type] || DEFAULT_STAGE
      @jobs[name] = { stage: stage }.merge(job)
90 91
    end

D
Douwe Maan 已提交
92 93
    def build_job(name, job)
      {
K
Kamil Trzcinski 已提交
94
        stage_idx: stages.index(job[:stage]),
D
Douwe Maan 已提交
95
        stage: job[:stage],
96
        commands: [job[:before_script] || [@before_script], job[:script]].flatten.compact.join("\n"),
K
Kamil Trzcinski 已提交
97
        tag_list: job[:tags] || [],
D
Douwe Maan 已提交
98 99 100 101
        name: name,
        only: job[:only],
        except: job[:except],
        allow_failure: job[:allow_failure] || false,
102
        when: job[:when] || 'on_success',
103
        environment: job[:environment],
D
Douwe Maan 已提交
104 105
        options: {
          image: job[:image] || @image,
K
Kamil Trzcinski 已提交
106
          services: job[:services] || @services,
107 108
          artifacts: job[:artifacts],
          cache: job[:cache] || @cache,
109
          dependencies: job[:dependencies],
110
          after_script: job[:after_script] || @after_script,
D
Douwe Maan 已提交
111 112 113 114 115
        }.compact
      }
    end

    def validate!
K
Kamil Trzcinski 已提交
116 117 118 119 120 121 122 123 124 125
      validate_global!

      @jobs.each do |name, job|
        validate_job!(name, job)
      end

      true
    end

    def validate_global!
126 127
      unless @after_script.nil? || validate_array_of_strings(@after_script)
        raise ValidationError, "after_script should be an array of strings"
128 129
      end

D
Douwe Maan 已提交
130 131 132 133 134 135 136 137 138
      unless @services.nil? || validate_array_of_strings(@services)
        raise ValidationError, "services should be an array of strings"
      end

      unless @stages.nil? || validate_array_of_strings(@stages)
        raise ValidationError, "stages should be an array of strings"
      end

      unless @variables.nil? || validate_variables(@variables)
G
Grzegorz Bizon 已提交
139
        raise ValidationError, "variables should be a map of key-value strings"
D
Douwe Maan 已提交
140 141
      end

K
Kamil Trzcinski 已提交
142 143
      validate_global_cache! if @cache
    end
144

K
Kamil Trzcinski 已提交
145
    def validate_global_cache!
K
Kamil Trzcinski 已提交
146 147 148 149 150 151
      @cache.keys.each do |key|
        unless ALLOWED_CACHE_KEYS.include? key
          raise ValidationError, "#{name} cache unknown parameter #{key}"
        end
      end

K
Kamil Trzcinski 已提交
152 153
      if @cache[:key] && !validate_string(@cache[:key])
        raise ValidationError, "cache:key parameter should be a string"
154 155
      end

K
Kamil Trzcinski 已提交
156 157
      if @cache[:untracked] && !validate_boolean(@cache[:untracked])
        raise ValidationError, "cache:untracked parameter should be an boolean"
D
Douwe Maan 已提交
158 159
      end

K
Kamil Trzcinski 已提交
160 161 162
      if @cache[:paths] && !validate_array_of_strings(@cache[:paths])
        raise ValidationError, "cache:paths parameter should be an array of strings"
      end
D
Douwe Maan 已提交
163 164 165
    end

    def validate_job!(name, job)
166 167 168
      validate_job_name!(name)
      validate_job_keys!(name, job)
      validate_job_types!(name, job)
K
Kamil Trzcinski 已提交
169
      validate_job_script!(name, job)
170 171

      validate_job_stage!(name, job) if job[:stage]
172
      validate_job_variables!(name, job) if job[:variables]
173 174
      validate_job_cache!(name, job) if job[:cache]
      validate_job_artifacts!(name, job) if job[:artifacts]
175
      validate_job_dependencies!(name, job) if job[:dependencies]
176 177 178
    end

    def validate_job_name!(name)
K
Kamil Trzcinski 已提交
179 180 181
      if name.blank? || !validate_string(name)
        raise ValidationError, "job name should be non-empty string"
      end
182
    end
K
Kamil Trzcinski 已提交
183

184
    def validate_job_keys!(name, job)
D
Douwe Maan 已提交
185 186
      job.keys.each do |key|
        unless ALLOWED_JOB_KEYS.include? key
K
Kamil Trzcinski 已提交
187
          raise ValidationError, "#{name} job: unknown parameter #{key}"
D
Douwe Maan 已提交
188 189
        end
      end
190
    end
D
Douwe Maan 已提交
191

192
    def validate_job_types!(name, job)
K
Kamil Trzcinski 已提交
193 194
      if job[:image] && !validate_string(job[:image])
        raise ValidationError, "#{name} job: image should be a string"
D
Douwe Maan 已提交
195 196 197
      end

      if job[:services] && !validate_array_of_strings(job[:services])
K
Kamil Trzcinski 已提交
198
        raise ValidationError, "#{name} job: services should be an array of strings"
D
Douwe Maan 已提交
199 200 201
      end

      if job[:tags] && !validate_array_of_strings(job[:tags])
K
Kamil Trzcinski 已提交
202
        raise ValidationError, "#{name} job: tags parameter should be an array of strings"
D
Douwe Maan 已提交
203 204
      end

205 206
      if job[:only] && !validate_array_of_strings_or_regexps(job[:only])
        raise ValidationError, "#{name} job: only parameter should be an array of strings or regexps"
D
Douwe Maan 已提交
207 208
      end

209 210
      if job[:except] && !validate_array_of_strings_or_regexps(job[:except])
        raise ValidationError, "#{name} job: except parameter should be an array of strings or regexps"
D
Douwe Maan 已提交
211 212
      end

213 214
      if job[:allow_failure] && !validate_boolean(job[:allow_failure])
        raise ValidationError, "#{name} job: allow_failure parameter should be an boolean"
215 216
      end

K
Kamil Trzcinski 已提交
217
      if job[:when] && !job[:when].in?(%w[on_success on_failure always])
218 219
        raise ValidationError, "#{name} job: when parameter should be on_success, on_failure or always"
      end
220

221 222
      if job[:environment] && !validate_environment(job[:environment])
        raise ValidationError, "#{name} job: environment parameter #{Gitlab::Regex.environment_name_regex_message}"
223
      end
224
    end
225

K
Kamil Trzcinski 已提交
226 227 228 229 230 231 232 233 234 235 236 237 238 239
    def validate_job_script!(name, job)
      if !validate_string(job[:script]) && !validate_array_of_strings(job[:script])
        raise ValidationError, "#{name} job: script should be a string or an array of a strings"
      end

      if job[:before_script] && !validate_array_of_strings(job[:before_script])
        raise ValidationError, "#{name} job: before_script should be an array of strings"
      end

      if job[:after_script] && !validate_array_of_strings(job[:after_script])
        raise ValidationError, "#{name} job: after_script should be an array of strings"
      end
    end

240 241 242
    def validate_job_stage!(name, job)
      unless job[:stage].is_a?(String) && job[:stage].in?(stages)
        raise ValidationError, "#{name} job: stage parameter should be #{stages.join(", ")}"
K
Kamil Trzcinski 已提交
243
      end
244
    end
K
Kamil Trzcinski 已提交
245

246
    def validate_job_variables!(name, job)
G
Grzegorz Bizon 已提交
247
      unless validate_variables(job[:variables])
248
        raise ValidationError,
G
Grzegorz Bizon 已提交
249
          "#{name} job: variables should be a map of key-value strings"
250 251 252
      end
    end

253
    def validate_job_cache!(name, job)
K
Kamil Trzcinski 已提交
254 255 256 257 258 259
      job[:cache].keys.each do |key|
        unless ALLOWED_CACHE_KEYS.include? key
          raise ValidationError, "#{name} job: cache unknown parameter #{key}"
        end
      end

260 261 262 263
      if job[:cache][:key] && !validate_string(job[:cache][:key])
        raise ValidationError, "#{name} job: cache:key parameter should be a string"
      end

264 265
      if job[:cache][:untracked] && !validate_boolean(job[:cache][:untracked])
        raise ValidationError, "#{name} job: cache:untracked parameter should be an boolean"
D
Douwe Maan 已提交
266
      end
267

268 269
      if job[:cache][:paths] && !validate_array_of_strings(job[:cache][:paths])
        raise ValidationError, "#{name} job: cache:paths parameter should be an array of strings"
270
      end
D
Douwe Maan 已提交
271 272
    end

273
    def validate_job_artifacts!(name, job)
K
Kamil Trzcinski 已提交
274 275 276 277 278 279
      job[:artifacts].keys.each do |key|
        unless ALLOWED_ARTIFACTS_KEYS.include? key
          raise ValidationError, "#{name} job: artifacts unknown parameter #{key}"
        end
      end

280 281 282 283
      if job[:artifacts][:name] && !validate_string(job[:artifacts][:name])
        raise ValidationError, "#{name} job: artifacts:name parameter should be a string"
      end

284 285 286 287 288 289 290
      if job[:artifacts][:untracked] && !validate_boolean(job[:artifacts][:untracked])
        raise ValidationError, "#{name} job: artifacts:untracked parameter should be an boolean"
      end

      if job[:artifacts][:paths] && !validate_array_of_strings(job[:artifacts][:paths])
        raise ValidationError, "#{name} job: artifacts:paths parameter should be an array of strings"
      end
K
Kamil Trzcinski 已提交
291

K
Kamil Trzcinski 已提交
292
      if job[:artifacts][:when] && !job[:artifacts][:when].in?(%w[on_success on_failure always])
K
Kamil Trzcinski 已提交
293 294
        raise ValidationError, "#{name} job: artifacts:when parameter should be on_success, on_failure or always"
      end
295 296 297 298

      if job[:artifacts][:expire_in] && !validate_duration(job[:artifacts][:expire_in])
        raise ValidationError, "#{name} job: artifacts:expire_in parameter should be a duration"
      end
299
    end
D
Douwe Maan 已提交
300

301
    def validate_job_dependencies!(name, job)
302
      unless validate_array_of_strings(job[:dependencies])
303 304 305 306 307 308
        raise ValidationError, "#{name} job: dependencies parameter should be an array of strings"
      end

      stage_index = stages.index(job[:stage])

      job[:dependencies].each do |dependency|
309
        raise ValidationError, "#{name} job: undefined dependency: #{dependency}" unless @jobs[dependency.to_sym]
310

311
        unless stages.index(@jobs[dependency.to_sym][:stage]) < stage_index
312 313 314 315 316
          raise ValidationError, "#{name} job: dependency #{dependency} is not defined in prior stages"
        end
      end
    end

317
    def process?(only_params, except_params, ref, tag, trigger_request)
318
      if only_params.present?
319
        return false unless matching?(only_params, ref, tag, trigger_request)
320 321 322
      end

      if except_params.present?
323
        return false if matching?(except_params, ref, tag, trigger_request)
324 325 326 327 328
      end

      true
    end

329
    def matching?(patterns, ref, tag, trigger_request)
330
      patterns.any? do |pattern|
331
        match_ref?(pattern, ref, tag, trigger_request)
332 333 334
      end
    end

J
Jason Roehm 已提交
335
    def match_ref?(pattern, ref, tag, trigger_request)
336 337 338 339
      pattern, path = pattern.split('@', 2)
      return false if path && path != self.path
      return true if tag && pattern == 'tags'
      return true if !tag && pattern == 'branches'
J
Jason Roehm 已提交
340
      return true if trigger_request.present? && pattern == 'triggers'
341 342 343 344 345 346 347

      if pattern.first == "/" && pattern.last == "/"
        Regexp.new(pattern[1...-1]) =~ ref
      else
        pattern == ref
      end
    end
D
Douwe Maan 已提交
348
  end
349
end