summaryrefslogtreecommitdiff
path: root/app/workers/issue_rebalancing_worker.rb
blob: 13e02c37bdb31d5aaacf0a16ab9b87ed2270bac9 (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
# frozen_string_literal: true

class IssueRebalancingWorker
  include ApplicationWorker

  data_consistency :always

  sidekiq_options retry: 3

  idempotent!
  urgency :low
  feature_category :issue_tracking
  tags :exclude_from_kubernetes
  deduplicate :until_executed, including_scheduled: true

  def perform(ignore = nil, project_id = nil, root_namespace_id = nil)
    # we need to have exactly one of the project_id and root_namespace_id params be non-nil
    raise ArgumentError, "Expected only one of the params project_id: #{project_id} and root_namespace_id: #{root_namespace_id}" if project_id && root_namespace_id
    return if project_id.nil? && root_namespace_id.nil?

    # pull the projects collection to be rebalanced either the project if namespace is not a group(i.e. user namesapce)
    # or the root namespace, this also makes the worker backward compatible with previous version where a project_id was
    # passed as the param
    projects_to_rebalance = projects_collection(project_id, root_namespace_id)

    # something might have happened with the namespace between scheduling the worker and actually running it,
    # maybe it was removed.
    if projects_to_rebalance.blank?
      Gitlab::ErrorTracking.log_exception(
        ArgumentError.new("Projects to be rebalanced not found for arguments: project_id #{project_id}, root_namespace_id: #{root_namespace_id}"),
        { project_id: project_id, root_namespace_id: root_namespace_id })

      return
    end

    # Temporary disable rebalancing for performance reasons
    # For more information check https://gitlab.com/gitlab-com/gl-infra/production/-/issues/4321
    return if projects_to_rebalance.take&.root_namespace&.issue_repositioning_disabled? # rubocop:disable CodeReuse/ActiveRecord

    IssueRebalancingService.new(projects_to_rebalance).execute
  rescue IssueRebalancingService::TooManyIssues => e
    Gitlab::ErrorTracking.log_exception(e, root_namespace_id: root_namespace_id, project_id: project_id)
  end

  private

  def projects_collection(project_id, root_namespace_id)
    # we can have either project_id(older version) or project_id if project is part of a user namespace and not a group
    # or root_namespace_id(newer version) never both.
    return Project.id_in([project_id]) if project_id

    Namespace.find_by_id(root_namespace_id)&.all_projects
  end
end