Module: Sequel::Postgres::PgAdvisoryLock
- Defined in:
- lib/sequel/extensions/pg_advisory_lock.rb
Defined Under Namespace
Classes: LockAlreadyRegistered
Constant Summary
collapse
- SESSION_LEVEL_LOCKS =
[
:pg_advisory_lock,
:pg_try_advisory_lock
].freeze
- TRANSACTION_LEVEL_LOCKS =
[
:pg_advisory_xact_lock,
:pg_try_advisory_xact_lock
].freeze
- LOCK_FUNCTIONS =
(SESSION_LEVEL_LOCKS + TRANSACTION_LEVEL_LOCKS).freeze
- DEFAULT_LOCK_FUNCTION =
:pg_advisory_lock
- UNLOCK_FUNCTION =
:pg_advisory_unlock
Instance Method Summary
collapse
Instance Method Details
#advisory_lock_key_for(lock_name) ⇒ Object
94
95
96
|
# File 'lib/sequel/extensions/pg_advisory_lock.rb', line 94
def advisory_lock_key_for(lock_name)
Zlib.crc32(lock_name.to_s) % 2 ** 31
end
|
#register_advisory_lock(name, lock_function = DEFAULT_LOCK_FUNCTION) ⇒ Object
Beth: not sure how much extra value this registration provides. It turns the name into a number, and makes sure the name/number is unique, and that you don’t try and use a different lock function with the same name.
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
|
# File 'lib/sequel/extensions/pg_advisory_lock.rb', line 73
def register_advisory_lock(name, lock_function = DEFAULT_LOCK_FUNCTION)
name = name.to_sym
if registered_advisory_locks.key?(name) && registered_advisory_locks[name][:lock_function] != lock_function
raise LockAlreadyRegistered, "Lock with name :#{name} is already registered with a different lock function (#{registered_advisory_locks[name][:lock_function]})"
end
key = advisory_lock_key_for(name)
name_for_key = registered_advisory_locks.keys.find { |n| registered_advisory_locks[n].fetch(:key) == key }
if name_for_key && name_for_key != name
raise Error, "Lock key #{key} is already taken"
end
function = lock_function.to_sym
unless LOCK_FUNCTIONS.include?(function)
raise Error, "Invalid lock function :#{function}"
end
registered_advisory_locks[name] = { key: key, lock_function: function }
end
|
#registered_advisory_locks ⇒ Object
32
33
34
|
# File 'lib/sequel/extensions/pg_advisory_lock.rb', line 32
def registered_advisory_locks
@registered_advisory_locks ||= Concurrent::Hash.new
end
|
#with_advisory_lock(name, id = nil) ⇒ Object
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
|
# File 'lib/sequel/extensions/pg_advisory_lock.rb', line 36
def with_advisory_lock(name, id = nil)
options = registered_advisory_locks.fetch(name.to_sym)
lock_key = options.fetch(:key)
function_params = [lock_key, id].compact
lock_function = options.fetch(:lock_function)
transaction_level_lock = TRANSACTION_LEVEL_LOCKS.include?(lock_function)
if transaction_level_lock
unless in_transaction?
raise Error, "Transaction must be manually opened before using transaction level lock '#{lock_function}'"
end
if get(Sequel.function(lock_function, *function_params))
yield
end
else
synchronize do
if get(Sequel.function(lock_function, *function_params))
begin
result = yield
ensure
get(Sequel.function(UNLOCK_FUNCTION, *function_params))
result
end
end
end
end
end
|