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

    class DestroyError < StandardError; end

    DELETED_FLAG = '+deleted'

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
      project.team.truncate
D
Dmitriy Zaporozhets 已提交
21

22 23 24
      repo_path = project.path_with_namespace
      wiki_path = repo_path + '.wiki'

25 26 27 28 29
      # 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)

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

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

37 38 39 40 41 42 43 44 45
        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

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

51
    private
D
Dmitriy Zaporozhets 已提交
52

53
    def remove_repository(path)
54 55 56 57
      # 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
58
      return true unless gitlab_shell.exists?(project.repository_storage_path, path + '.git')
59 60 61

      new_path = removal_path(path)

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

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

      project.container_registry_repository.delete_tags
74 75
    end

76 77 78 79 80 81 82 83 84 85 86 87
    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 已提交
88
    end
89 90

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

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