summaryrefslogtreecommitdiff
path: root/app/services/ci/runners/set_runner_associated_projects_service.rb
blob: 5e33fdae2f47c542eb071bf625111ec3a04fc8b1 (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
# frozen_string_literal: true

module Ci
  module Runners
    class SetRunnerAssociatedProjectsService
      # @param [Ci::Runner] runner: the project runner to assign/unassign projects from
      # @param [User] current_user: the user performing the operation
      # @param [Array<Integer>] project_ids: the IDs of the associated projects to assign the runner to
      def initialize(runner:, current_user:, project_ids:)
        @runner = runner
        @current_user = current_user
        @project_ids = project_ids
      end

      def execute
        unless current_user&.can?(:assign_runner, runner)
          return ServiceResponse.error(message: 'user not allowed to assign runner', http_status: :forbidden)
        end

        return ServiceResponse.success if project_ids.nil?

        set_associated_projects
      end

      private

      def set_associated_projects
        new_project_ids = [runner.owner_project.id] + project_ids

        response = ServiceResponse.success
        runner.transaction do
          # rubocop:disable CodeReuse/ActiveRecord
          current_project_ids = runner.projects.ids
          # rubocop:enable CodeReuse/ActiveRecord

          unless associate_new_projects(new_project_ids, current_project_ids)
            response = ServiceResponse.error(message: 'failed to assign projects to runner')
            raise ActiveRecord::Rollback, response.errors
          end

          unless disassociate_old_projects(new_project_ids, current_project_ids)
            response = ServiceResponse.error(message: 'failed to destroy runner project')
            raise ActiveRecord::Rollback, response.errors
          end
        end

        response
      end

      def associate_new_projects(new_project_ids, current_project_ids)
        missing_projects = Project.id_in(new_project_ids - current_project_ids)
        missing_projects.all? { |project| runner.assign_to(project, current_user) }
      end

      def disassociate_old_projects(new_project_ids, current_project_ids)
        projects_to_be_deleted = current_project_ids - new_project_ids
        return true if projects_to_be_deleted.empty?

        Ci::RunnerProject
          .destroy_by(project_id: projects_to_be_deleted)
          .all?(&:destroyed?)
      end

      attr_reader :runner, :current_user, :project_ids
    end
  end
end

Ci::Runners::SetRunnerAssociatedProjectsService.prepend_mod