summaryrefslogtreecommitdiff
path: root/app/workers/hashed_storage/project_rollback_worker.rb
blob: ace9fea98a6a09f9c5e9bba470bd12f051761bf7 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
# frozen_string_literal: true

module HashedStorage
  class ProjectRollbackWorker
    include ApplicationWorker

    LEASE_TIMEOUT = 30.seconds.to_i

    queue_namespace :hashed_storage

    # rubocop: disable CodeReuse/ActiveRecord
    def perform(project_id, old_disk_path = nil)
      uuid = lease_for(project_id).try_obtain

      if uuid
        project = Project.find_by(id: project_id)
        return if project.nil? || project.pending_delete?

        old_disk_path ||= project.disk_path

        ::Projects::HashedStorage::RollbackService.new(project, old_disk_path, logger: logger).execute
      else
        return false
      end

    ensure
      cancel_lease_for(project_id, uuid) if uuid
    end

    # rubocop: enable CodeReuse/ActiveRecord

    def lease_for(project_id)
      Gitlab::ExclusiveLease.new(lease_key(project_id), timeout: LEASE_TIMEOUT)
    end

    private

    def lease_key(project_id)
      # we share the same lease key for both migration and rollback so they don't run simultaneously
      "#{ProjectMigrateWorker::LEASE_KEY_SEGMENT}:#{project_id}"
    end

    def cancel_lease_for(project_id, uuid)
      Gitlab::ExclusiveLease.cancel(lease_key(project_id), uuid)
    end
  end
end