transfer_service.rb 5.2 KB
Newer Older
1 2
# frozen_string_literal: true

3 4 5 6 7 8 9 10
# Projects::TransferService class
#
# Used for transfer project to another namespace
#
# Ex.
#   # Move projects to namespace with ID 17 by user
#   Projects::TransferService.new(project, user, namespace_id: 17).execute
#
11
module Projects
12
  class TransferService < BaseService
13
    include Gitlab::ShellAdapter
14
    TransferError = Class.new(StandardError)
15

16
    def execute(new_namespace)
17 18 19
      @new_namespace = new_namespace

      if @new_namespace.blank?
20
        raise TransferError, s_('TransferProject|Please select a new namespace for your project.')
21
      end
22 23

      unless allowed_transfer?(current_user, project)
24
        raise TransferError, s_('TransferProject|Transfer failed, please contact an admin.')
25
      end
26 27 28

      transfer(project)

A
Andreas Brandl 已提交
29 30
      current_user.invalidate_personal_projects_count

31
      true
32
    rescue Projects::TransferService::TransferError => ex
33
      project.reset
34
      project.errors.add(:new_namespace, ex.message)
35 36
      false
    end
37

38
    private
39

40
    # rubocop: disable CodeReuse/ActiveRecord
41
    def transfer(project)
42
      @old_path = project.full_path
43 44 45
      @old_group = project.group
      @new_path = File.join(@new_namespace.try(:full_path) || '', project.path)
      @old_namespace = project.namespace
46

J
James Lopez 已提交
47
      if Project.where(namespace_id: @new_namespace.try(:id)).where('path = ? or name = ?', project.path, project.name).exists?
48
        raise TransferError.new(s_("TransferProject|Project with same name or path in target namespace already exists"))
49
      end
50

51 52
      if project.has_container_registry_tags?
        # We currently don't support renaming repository if it contains tags in container registry
53
        raise TransferError.new(s_('TransferProject|Project cannot be transferred, because tags are present in its container registry'))
54
      end
55

56 57
      attempt_transfer_transaction
    end
58
    # rubocop: enable CodeReuse/ActiveRecord
59 60 61 62

    def attempt_transfer_transaction
      Project.transaction do
        project.expire_caches_before_rename(@old_path)
63

64
        update_namespace_and_visibility(@new_namespace)
65

V
Valery Sizov 已提交
66
        # Notifications
67
        project.send_move_instructions(@old_path)
V
Valery Sizov 已提交
68

69 70
        # Directories on disk
        move_project_folders(project)
71

72
        # Move missing group labels to project
73
        Labels::TransferService.new(current_user, @old_group, project).execute
74

75 76 77
        # Move missing group milestones
        Milestones::TransferService.new(current_user, @old_group, project).execute

78
        # Move uploads
79
        move_project_uploads(project)
80

81
        # Move pages
82
        Gitlab::PagesTransfer.new.move_project(project.path, @old_namespace.full_path, @new_namespace.full_path)
83

84
        project.old_path_with_namespace = @old_path
85

86
        update_repository_configuration(@new_path)
87

88
        execute_system_hooks
89
      end
90 91 92 93 94
    rescue Exception # rubocop:disable Lint/RescueException
      rollback_side_effects
      raise
    ensure
      refresh_permissions
95 96
    end

97 98
    def allowed_transfer?(current_user, project)
      @new_namespace &&
99
        can?(current_user, :change_namespace, project) &&
100 101
        @new_namespace.id != project.namespace_id &&
        current_user.can?(:create_projects, @new_namespace)
102
    end
103

104 105 106 107 108 109 110
    def update_namespace_and_visibility(to_namespace)
      # Apply new namespace id and visibility level
      project.namespace = to_namespace
      project.visibility_level = to_namespace.visibility_level unless project.visibility_level_allowed_by_group?
      project.save!
    end

111
    def update_repository_configuration(full_path)
112
      project.write_repository_config(gl_full_path: full_path)
113
      project.track_project_repository
114 115
    end

116
    def refresh_permissions
117 118
      # This ensures we only schedule 1 job for every user that has access to
      # the namespaces.
119 120
      user_ids = @old_namespace.user_ids_for_project_authorizations |
        @new_namespace.user_ids_for_project_authorizations
121 122 123

      UserProjectAccessChangedService.new(user_ids).execute
    end
124 125 126

    def rollback_side_effects
      rollback_folder_move
127
      project.reset
128
      update_namespace_and_visibility(@old_namespace)
129
      update_repository_configuration(@old_path)
130 131 132 133 134 135 136 137
    end

    def rollback_folder_move
      move_repo_folder(@new_path, @old_path)
      move_repo_folder("#{@new_path}.wiki", "#{@old_path}.wiki")
    end

    def move_repo_folder(from_name, to_name)
138
      gitlab_shell.mv_repository(project.repository_storage, from_name, to_name)
139 140 141 142 143
    end

    def execute_system_hooks
      SystemHooksService.new.execute_hooks_for(project, :transfer)
    end
144 145 146 147 148 149

    def move_project_folders(project)
      return if project.hashed_storage?(:repository)

      # Move main repository
      unless move_repo_folder(@old_path, @new_path)
150
        raise TransferError.new(s_("TransferProject|Cannot move project"))
151 152 153 154 155 156 157 158
      end

      # Disk path is changed; we need to ensure we reload it
      project.reload_repository!

      # Move wiki repo also if present
      move_repo_folder("#{@old_path}.wiki", "#{@new_path}.wiki")
    end
159 160 161 162 163 164 165 166 167 168

    def move_project_uploads(project)
      return if project.hashed_storage?(:attachments)

      Gitlab::UploadsTransfer.new.move_project(
        project.path,
        @old_namespace.full_path,
        @new_namespace.full_path
      )
    end
169 170
  end
end