git_garbage_collect_worker.rb 1.68 KB
Newer Older
1 2
class GitGarbageCollectWorker
  include Sidekiq::Worker
3
  include DedicatedSidekiqQueue
4
  include Gitlab::CurrentSettings
5

6
  sidekiq_options retry: false
7

8
  def perform(project_id, task = :gc, lease_key = nil, lease_uuid = nil)
9
    project = Project.find(project_id)
10 11 12 13 14 15 16 17 18 19
    task = task.to_sym

    cmd = command(task)
    repo_path = project.repository.path_to_repo
    description = "'#{cmd.join(' ')}' in #{repo_path}"

    Gitlab::GitLogger.info(description)

    output, status = Gitlab::Popen.popen(cmd, repo_path)
    Gitlab::GitLogger.error("#{description} failed:\n#{output}") unless status.zero?
20

21
    # Refresh the branch cache in case garbage collection caused a ref lookup to fail
22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44
    flush_ref_caches(project) if task == :gc
  ensure
    Gitlab::ExclusiveLease.cancel(lease_key, lease_uuid) if lease_key.present? && lease_uuid.present?
  end

  private

  def command(task)
    case task
    when :gc
      git(write_bitmaps: bitmaps_enabled?) + %w[gc]
    when :full_repack
      git(write_bitmaps: bitmaps_enabled?) + %w[repack -A -d --pack-kept-objects]
    when :incremental_repack
      # Normal git repack fails when bitmaps are enabled. It is impossible to
      # create a bitmap here anyway.
      git(write_bitmaps: false) + %w[repack -d]
    else
      raise "Invalid gc task: #{task.inspect}"
    end
  end

  def flush_ref_caches(project)
45
    project.repository.after_create_branch
46 47
    project.repository.branch_names
    project.repository.has_visible_content?
48
  end
49 50 51 52 53 54 55 56 57

  def bitmaps_enabled?
    current_application_settings.housekeeping_bitmaps_enabled
  end

  def git(write_bitmaps:)
    config_value = write_bitmaps ? 'true' : 'false'
    %W[git -c repack.writeBitmaps=#{config_value}]
  end
58
end