destroy_service.rb 3.1 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
      Projects::UnlinkForkService.new(project, current_user).execute

32 33
      Project.transaction do
        project.destroy!
D
Dmitriy Zaporozhets 已提交
34

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

39 40 41 42 43 44 45 46 47
        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

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

53
    private
D
Dmitriy Zaporozhets 已提交
54

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

      new_path = removal_path(path)

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

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

      project.container_registry_repository.delete_tags
76 77
    end

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

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

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