summaryrefslogtreecommitdiff
path: root/app/services/ci/queue/build_queue_service.rb
blob: 3276c427923f4e66d583620e1969dee456eebbc1 (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
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
# frozen_string_literal: true

module Ci
  module Queue
    class BuildQueueService
      include ::Gitlab::Utils::StrongMemoize

      attr_reader :runner

      def initialize(runner)
        @runner = runner
      end

      def new_builds
        strategy.new_builds
      end

      ##
      # This is overridden in EE
      #
      def builds_for_shared_runner
        strategy.builds_for_shared_runner
      end

      # rubocop:disable CodeReuse/ActiveRecord
      def builds_for_group_runner
        if strategy.use_denormalized_namespace_traversal_ids?
          strategy.builds_for_group_runner
        else
          # Workaround for weird Rails bug, that makes `runner.groups.to_sql` to return `runner_id = NULL`
          groups = ::Group.joins(:runner_namespaces).merge(runner.runner_namespaces)

          hierarchy_groups = Gitlab::ObjectHierarchy
            .new(groups)
            .base_and_descendants

          projects = Project.where(namespace_id: hierarchy_groups)
            .with_group_runners_enabled
            .with_builds_enabled
            .without_deleted

          relation = new_builds.where(project: projects)

          order(relation)
        end
      end

      def builds_for_project_runner
        relation = new_builds
          .where(project: runner_projects_relation)

        order(relation)
      end

      def builds_queued_before(relation, time)
        relation.queued_before(time)
      end

      def builds_for_protected_runner(relation)
        relation.ref_protected
      end

      def builds_matching_tag_ids(relation, ids)
        strategy.builds_matching_tag_ids(relation, ids)
      end

      def builds_with_any_tags(relation)
        strategy.builds_with_any_tags(relation)
      end

      def order(relation)
        strategy.order(relation)
      end

      def execute(relation)
        strategy.build_ids(relation)
      end

      private

      def strategy
        strong_memoize(:strategy) do
          if ::Feature.enabled?(:ci_pending_builds_queue_source, runner, default_enabled: :yaml)
            Queue::PendingBuildsStrategy.new(runner)
          else
            Queue::BuildsTableStrategy.new(runner)
          end
        end
      end

      def runner_projects_relation
        if ::Feature.enabled?(:ci_pending_builds_project_runners_decoupling, runner, default_enabled: :yaml)
          runner.runner_projects.select(:project_id)
        else
          runner.projects.without_deleted.with_builds_enabled
        end
      end
    end
  end
end

Ci::Queue::BuildQueueService.prepend_mod_with('Ci::Queue::BuildQueueService')