summaryrefslogtreecommitdiff
path: root/app/services/ci/after_requeue_job_service.rb
blob: bc70dd3bea4e757fe5620c776fcb42aecf1c1484 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
# frozen_string_literal: true

module Ci
  class AfterRequeueJobService < ::BaseService
    def execute(processable)
      @processable = processable

      process_subsequent_jobs
      reset_source_bridge
    end

    private

    def process_subsequent_jobs
      dependent_jobs.each do |job|
        process(job)
      end
    end

    def reset_source_bridge
      @processable.pipeline.reset_source_bridge!(current_user)
    end

    def dependent_jobs
      dependent_jobs = stage_dependent_jobs
        .or(needs_dependent_jobs)
        .ordered_by_stage

      if ::Feature.enabled?(:ci_fix_order_of_subsequent_jobs, @processable.pipeline.project, default_enabled: :yaml)
        dependent_jobs = ordered_by_dag(dependent_jobs)
      end

      dependent_jobs
    end

    def process(job)
      Gitlab::OptimisticLocking.retry_lock(job, name: 'ci_requeue_job') do |job|
        job.process(current_user)
      end
    end

    def stage_dependent_jobs
      skipped_jobs.after_stage(@processable.stage_idx)
    end

    def needs_dependent_jobs
      skipped_jobs.scheduling_type_dag.with_needs([@processable.name])
    end

    def skipped_jobs
      @skipped_jobs ||= @processable.pipeline.processables.skipped
    end

    # rubocop: disable CodeReuse/ActiveRecord
    def ordered_by_dag(jobs)
      sorted_job_names = sort_jobs(jobs).each_with_index.to_h

      jobs.preload(:needs).group_by(&:stage_idx).flat_map do |_, stage_jobs|
        stage_jobs.sort_by { |job| sorted_job_names.fetch(job.name) }
      end
    end

    def sort_jobs(jobs)
      Gitlab::Ci::YamlProcessor::Dag.order(
        jobs.to_h do |job|
          [job.name, job.needs.map(&:name)]
        end
      )
    end
    # rubocop: enable CodeReuse/ActiveRecord
  end
end