destroy_service.rb 3.1 KB
Newer Older
D
Dmitriy Zaporozhets 已提交
1
module Projects
2
  class DestroyService < BaseService
3 4 5 6
    include Gitlab::ShellAdapter

    class DestroyError < StandardError; end

D
Douwe Maan 已提交
7
    DELETED_FLAG = '+deleted'.freeze
8

S
Stan Hu 已提交
9 10 11 12 13 14
    def async_execute
      project.transaction do
        project.update_attribute(:pending_delete, true)
        job_id = ProjectDestroyWorker.perform_async(project.id, current_user.id, params)
        Rails.logger.info("User #{current_user.id} scheduled destruction of project #{project.path_with_namespace} with job ID #{job_id}")
      end
15 16
    end

17
    def execute
D
Dmitriy Zaporozhets 已提交
18 19
      return false unless can?(current_user, :remove_project, project)

20 21 22
      repo_path = project.path_with_namespace
      wiki_path = repo_path + '.wiki'

23 24 25 26 27
      # Flush the cache for both repositories. This has to be done _before_
      # removing the physical repositories as some expiration code depends on
      # Git data (e.g. a list of branch names).
      flush_caches(project, wiki_path)

28 29
      Projects::UnlinkForkService.new(project, current_user).execute

30
      Project.transaction do
31
        project.team.truncate
32
        project.destroy!
D
Dmitriy Zaporozhets 已提交
33

34
        unless remove_registry_tags
35
          raise_error('Failed to remove project container registry. Please try again or contact administrator')
36 37
        end

38 39 40 41 42 43 44 45 46
        unless remove_repository(repo_path)
          raise_error('Failed to remove project repository. Please try again or contact administrator')
        end

        unless remove_repository(wiki_path)
          raise_error('Failed to remove wiki repository. Please try again or contact administrator')
        end
      end

47
      log_info("Project \"#{project.path_with_namespace}\" was removed")
48 49 50
      system_hook_service.execute_hooks_for(project, :destroy)
      true
    end
D
Dmitriy Zaporozhets 已提交
51

52
    private
D
Dmitriy Zaporozhets 已提交
53

54
    def remove_repository(path)
55 56 57 58
      # Skip repository removal. We use this flag when remove user or group
      return true if params[:skip_repo] == true

      # There is a possibility project does not have repository or wiki
59
      return true unless gitlab_shell.exists?(project.repository_storage_path, path + '.git')
60 61 62

      new_path = removal_path(path)

63
      if gitlab_shell.mv_repository(project.repository_storage_path, path, new_path)
64
        log_info("Repository \"#{path}\" moved to \"#{new_path}\"")
65
        GitlabShellWorker.perform_in(5.minutes, :remove_repository, project.repository_storage_path, new_path)
66 67 68 69 70
      else
        false
      end
    end

71
    def remove_registry_tags
72
      return true unless Gitlab.config.registry.enabled
K
Kamil Trzcinski 已提交
73 74

      project.container_registry_repository.delete_tags
75 76
    end

77 78 79 80 81 82 83 84 85 86 87 88
    def raise_error(message)
      raise DestroyError.new(message)
    end

    # Build a path for removing repositories
    # We use `+` because its not allowed by GitLab so user can not create
    # project with name cookies+119+deleted and capture someone stalled repository
    #
    # gitlab/cookies.git -> gitlab/cookies+119+deleted.git
    #
    def removal_path(path)
      "#{path}+#{project.id}#{DELETED_FLAG}"
D
Dmitriy Zaporozhets 已提交
89
    end
90 91

    def flush_caches(project, wiki_path)
92
      project.repository.before_delete
93

94
      Repository.new(wiki_path, project).before_delete
95
    end
D
Dmitriy Zaporozhets 已提交
96 97
  end
end