process_pipeline_service.rb 2.5 KB
Newer Older
1 2 3 4 5 6 7
module Ci
  class ProcessPipelineService < BaseService
    attr_reader :pipeline

    def execute(pipeline)
      @pipeline = pipeline

8 9
      update_retried

K
Kamil Trzcinski 已提交
10 11 12 13
      new_builds =
        stage_indexes_of_created_builds.map do |index|
          process_stage(index)
        end
14

K
Kamil Trzcinski 已提交
15
      @pipeline.update_status
16

K
Kamil Trzcinski 已提交
17
      new_builds.flatten.any?
18 19 20 21 22 23 24
    end

    private

    def process_stage(index)
      current_status = status_for_prior_stages(index)

25 26
      return if HasStatus::BLOCKED_STATUS == current_status

K
Kamil Trzcinski 已提交
27 28
      if HasStatus::COMPLETED_STATUSES.include?(current_status)
        created_builds_in_stage(index).select do |build|
29 30
          Gitlab::OptimisticLocking.retry_lock(build) do |subject|
            process_build(subject, current_status)
K
Kamil Trzcinski 已提交
31
          end
32
        end
33 34 35 36 37
      end
    end

    def process_build(build, current_status)
      if valid_statuses_for_when(build.when).include?(current_status)
38
        build.action? ? build.actionize : build.enqueue
39 40
        true
      else
S
Stan Hu 已提交
41
        build.skip
42 43 44 45 46 47 48
        false
      end
    end

    def valid_statuses_for_when(value)
      case value
      when 'on_success'
49
        %w[success skipped]
50 51 52
      when 'on_failure'
        %w[failed]
      when 'always'
53
        %w[success failed skipped]
54
      when 'manual'
55
        %w[success skipped]
56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75
      else
        []
      end
    end

    def status_for_prior_stages(index)
      pipeline.builds.where('stage_idx < ?', index).latest.status || 'success'
    end

    def stage_indexes_of_created_builds
      created_builds.order(:stage_idx).pluck('distinct stage_idx')
    end

    def created_builds_in_stage(index)
      created_builds.where(stage_idx: index)
    end

    def created_builds
      pipeline.builds.created
    end
76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93

    # This method is for compatibility and data consistency and should be removed with 9.3 version of GitLab
    # This replicates what is db/post_migrate/20170416103934_upate_retried_for_ci_build.rb
    # and ensures that functionality will not be broken before migration is run
    # this updates only when there are data that needs to be updated, there are two groups with no retried flag
    def update_retried
      # find the latest builds for each name
      latest_statuses = pipeline.statuses.latest
        .group(:name)
        .having('count(*) > 1')
        .pluck('max(id)', 'name')

      # mark builds that are retried
      pipeline.statuses.latest
        .where(name: latest_statuses.map(&:second))
        .where.not(id: latest_statuses.map(&:first))
        .update_all(retried: true) if latest_statuses.any?
    end
94 95
  end
end