BigW Consortium Gitlab

gitlab_ci_yaml_processor.rb 10.6 KB
Newer Older
1 2
module Ci
  class GitlabCiYamlProcessor
3
    class ValidationError < StandardError; end
4 5 6

    DEFAULT_STAGES = %w(build test deploy)
    DEFAULT_STAGE = 'test'
7
    ALLOWED_YAML_KEYS = [:before_script, :after_script, :image, :services, :types, :stages, :variables, :cache]
8 9
    ALLOWED_JOB_KEYS = [:tags, :script, :only, :except, :type, :image, :services,
                        :allow_failure, :type, :stage, :when, :artifacts, :cache,
10
                        :dependencies, :before_script, :after_script, :variables]
11

12
    attr_reader :before_script, :after_script, :image, :services, :path, :cache
13

14
    def initialize(config, path = nil)
15
      @config = Gitlab::Ci::Config.new(config).to_hash
16
      @path = path
17 18 19 20

      initial_parsing

      validate!
21
    rescue Gitlab::Ci::Config::Loader::FormatError => e
22
      raise ValidationError, e.message
23 24
    end

25 26
    def builds_for_stage_and_ref(stage, ref, tag = false, trigger_request = nil)
      builds.select{|build| build[:stage] == stage && process?(build[:only], build[:except], ref, tag, trigger_request)}
27 28 29 30 31 32 33 34 35 36 37 38
    end

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

    def stages
      @stages || DEFAULT_STAGES
    end

39
    def global_variables
40 41 42 43
      @variables
    end

    def job_variables(name)
44 45 46 47
      job = @jobs[name.to_sym]
      return [] unless job

      job.fetch(:variables, [])
48 49
    end

50 51 52 53
    private

    def initial_parsing
      @before_script = @config[:before_script] || []
54
      @after_script = @config[:after_script]
55 56 57 58
      @image = @config[:image]
      @services = @config[:services]
      @stages = @config[:stages] || @config[:types]
      @variables = @config[:variables] || {}
59
      @cache = @config[:cache]
60
      @jobs = {}
61

62
      @config.except!(*ALLOWED_YAML_KEYS)
Tomasz Maczukin committed
63
      @config.each { |name, param| add_job(name, param) }
64

65
      raise ValidationError, "Please define at least one job" if @jobs.none?
66 67
    end

68 69 70 71 72 73 74
    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)
75 76
    end

77 78
    def build_job(name, job)
      {
79
        stage_idx: stages.index(job[:stage]),
80
        stage: job[:stage],
81
        commands: [job[:before_script] || @before_script, job[:script]].flatten.join("\n"),
Kamil Trzcinski committed
82
        tag_list: job[:tags] || [],
83 84 85 86
        name: name,
        only: job[:only],
        except: job[:except],
        allow_failure: job[:allow_failure] || false,
87
        when: job[:when] || 'on_success',
88 89
        options: {
          image: job[:image] || @image,
90
          services: job[:services] || @services,
91 92
          artifacts: job[:artifacts],
          cache: job[:cache] || @cache,
93
          dependencies: job[:dependencies],
94
          after_script: job[:after_script] || @after_script,
95 96 97 98 99
        }.compact
      }
    end

    def validate!
100 101 102 103 104 105 106 107 108 109
      validate_global!

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

      true
    end

    def validate_global!
110 111 112 113
      unless validate_array_of_strings(@before_script)
        raise ValidationError, "before_script should be an array of strings"
      end

114 115
      unless @after_script.nil? || validate_array_of_strings(@after_script)
        raise ValidationError, "after_script should be an array of strings"
116 117
      end

118 119 120 121 122 123 124 125 126 127 128 129 130
      unless @image.nil? || @image.is_a?(String)
        raise ValidationError, "image should be a string"
      end

      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)
131
        raise ValidationError, "variables should be a map of key-value strings"
132 133
      end

134 135
      validate_global_cache! if @cache
    end
136

137 138 139
    def validate_global_cache!
      if @cache[:key] && !validate_string(@cache[:key])
        raise ValidationError, "cache:key parameter should be a string"
140 141
      end

142 143
      if @cache[:untracked] && !validate_boolean(@cache[:untracked])
        raise ValidationError, "cache:untracked parameter should be an boolean"
144 145
      end

146 147 148
      if @cache[:paths] && !validate_array_of_strings(@cache[:paths])
        raise ValidationError, "cache:paths parameter should be an array of strings"
      end
149 150 151
    end

    def validate_job!(name, job)
152 153 154
      validate_job_name!(name)
      validate_job_keys!(name, job)
      validate_job_types!(name, job)
Kamil Trzcinski committed
155
      validate_job_script!(name, job)
156 157

      validate_job_stage!(name, job) if job[:stage]
158
      validate_job_variables!(name, job) if job[:variables]
159 160
      validate_job_cache!(name, job) if job[:cache]
      validate_job_artifacts!(name, job) if job[:artifacts]
161
      validate_job_dependencies!(name, job) if job[:dependencies]
162 163 164
    end

    def validate_job_name!(name)
165 166 167
      if name.blank? || !validate_string(name)
        raise ValidationError, "job name should be non-empty string"
      end
168
    end
169

170
    def validate_job_keys!(name, job)
171 172
      job.keys.each do |key|
        unless ALLOWED_JOB_KEYS.include? key
173
          raise ValidationError, "#{name} job: unknown parameter #{key}"
174 175
        end
      end
176
    end
177

178
    def validate_job_types!(name, job)
179 180
      if job[:image] && !validate_string(job[:image])
        raise ValidationError, "#{name} job: image should be a string"
181 182 183
      end

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

      if job[:tags] && !validate_array_of_strings(job[:tags])
188
        raise ValidationError, "#{name} job: tags parameter should be an array of strings"
189 190 191
      end

      if job[:only] && !validate_array_of_strings(job[:only])
192
        raise ValidationError, "#{name} job: only parameter should be an array of strings"
193 194 195
      end

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

199 200
      if job[:allow_failure] && !validate_boolean(job[:allow_failure])
        raise ValidationError, "#{name} job: allow_failure parameter should be an boolean"
201 202
      end

203 204 205 206
      if job[:when] && !job[:when].in?(%w(on_success on_failure always))
        raise ValidationError, "#{name} job: when parameter should be on_success, on_failure or always"
      end
    end
207

Kamil Trzcinski committed
208 209 210 211 212 213 214 215 216 217 218 219 220 221
    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

222 223 224
    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(", ")}"
225
      end
226
    end
227

228
    def validate_job_variables!(name, job)
229
      unless validate_variables(job[:variables])
230
        raise ValidationError,
231
          "#{name} job: variables should be a map of key-value strings"
232 233 234
      end
    end

235
    def validate_job_cache!(name, job)
236 237 238 239
      if job[:cache][:key] && !validate_string(job[:cache][:key])
        raise ValidationError, "#{name} job: cache:key parameter should be a string"
      end

240 241
      if job[:cache][:untracked] && !validate_boolean(job[:cache][:untracked])
        raise ValidationError, "#{name} job: cache:untracked parameter should be an boolean"
242
      end
243

244 245
      if job[:cache][:paths] && !validate_array_of_strings(job[:cache][:paths])
        raise ValidationError, "#{name} job: cache:paths parameter should be an array of strings"
246
      end
247 248
    end

249
    def validate_job_artifacts!(name, job)
250 251 252 253
      if job[:artifacts][:name] && !validate_string(job[:artifacts][:name])
        raise ValidationError, "#{name} job: artifacts:name parameter should be a string"
      end

254 255 256 257 258 259 260 261
      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
    end
262

263
    def validate_job_dependencies!(name, job)
264
      unless validate_array_of_strings(job[:dependencies])
265 266 267 268 269 270
        raise ValidationError, "#{name} job: dependencies parameter should be an array of strings"
      end

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

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

273
        unless stages.index(@jobs[dependency.to_sym][:stage]) < stage_index
274 275 276 277 278
          raise ValidationError, "#{name} job: dependency #{dependency} is not defined in prior stages"
        end
      end
    end

279
    def validate_array_of_strings(values)
280
      values.is_a?(Array) && values.all? { |value| validate_string(value) }
281 282 283
    end

    def validate_variables(variables)
284 285 286 287 288
      variables.is_a?(Hash) && variables.all? { |key, value| validate_string(key) && validate_string(value) }
    end

    def validate_string(value)
      value.is_a?(String) || value.is_a?(Symbol)
289
    end
290

291 292 293 294
    def validate_boolean(value)
      value.in?([true, false])
    end

295
    def process?(only_params, except_params, ref, tag, trigger_request)
296
      if only_params.present?
297
        return false unless matching?(only_params, ref, tag, trigger_request)
298 299 300
      end

      if except_params.present?
301
        return false if matching?(except_params, ref, tag, trigger_request)
302 303 304 305 306
      end

      true
    end

307
    def matching?(patterns, ref, tag, trigger_request)
308
      patterns.any? do |pattern|
309
        match_ref?(pattern, ref, tag, trigger_request)
310 311 312
      end
    end

313
    def match_ref?(pattern, ref, tag, trigger_request)
314 315 316 317
      pattern, path = pattern.split('@', 2)
      return false if path && path != self.path
      return true if tag && pattern == 'tags'
      return true if !tag && pattern == 'branches'
318
      return true if trigger_request.present? && pattern == 'triggers'
319 320 321 322 323 324 325

      if pattern.first == "/" && pattern.last == "/"
        Regexp.new(pattern[1...-1]) =~ ref
      else
        pattern == ref
      end
    end
326
  end
327
end