BigW Consortium Gitlab

exclusive_lease.rb 1.48 KB
Newer Older
1 2
require 'securerandom'

3 4 5 6 7
module Gitlab
  # This class implements an 'exclusive lease'. We call it a 'lease'
  # because it has a set expiry time. We call it 'exclusive' because only
  # one caller may obtain a lease for a given key at a time. The
  # implementation is intended to work across GitLab processes and across
8
  # servers. It is a cheap alternative to using SQL queries and updates:
9 10
  # you do not need to change the SQL schema to start using
  # ExclusiveLease.
Jacob Vosmaer committed
11
  #
12
  class ExclusiveLease
13
    LUA_CANCEL_SCRIPT = <<-EOS.freeze
14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29
      local key, uuid = KEYS[1], ARGV[1]
      if redis.call("get", key) == uuid then
        redis.call("del", key)
      end
    EOS

    def self.cancel(key, uuid)
      Gitlab::Redis.with do |redis|
        redis.eval(LUA_CANCEL_SCRIPT, keys: [redis_key(key)], argv: [uuid])
      end
    end

    def self.redis_key(key)
      "gitlab:exclusive_lease:#{key}"
    end

Jacob Vosmaer committed
30
    def initialize(key, timeout:)
31 32 33
      @redis_key = self.class.redis_key(key)
      @timeout = timeout
      @uuid = SecureRandom.uuid
34 35
    end

36
    # Try to obtain the lease. Return lease UUID on success,
37 38
    # false if the lease is already taken.
    def try_obtain
39
      # Performing a single SET is atomic
40
      Gitlab::Redis.with do |redis|
41
        redis.set(@redis_key, @uuid, nx: true, ex: @timeout) && @uuid
42
      end
43 44
    end

45 46 47
    # Returns true if the key for this lease is set.
    def exists?
      Gitlab::Redis.with do |redis|
48
        redis.exists(@redis_key)
49 50
      end
    end
51 52
  end
end