Atomic ExpiringLock

Stephen Paul Weber created

Change summary

lib/expiring_lock.rb     |  9 +++++----
test/test_low_balance.rb | 28 +++++++++-------------------
test/test_web.rb         | 24 +++++++++---------------
3 files changed, 23 insertions(+), 38 deletions(-)

Detailed changes

lib/expiring_lock.rb 🔗

@@ -7,11 +7,12 @@ class ExpiringLock
 	end
 
 	def with(els=nil)
-		REDIS.exists(@name).then do |exists|
-			next els&.call if exists == 1
+		REDIS.set(@name, Time.now, "EX", @expiry, "NX").then do |result|
+			next els&.call if result.nil?
 
-			EMPromise.resolve(yield).then do |rval|
-				REDIS.setex(@name, @expiry, Time.now).then { rval }
+			EMPromise.resolve(yield).catch do |err|
+				REDIS.del(@name)
+				EMPromise.reject(err)
 			end
 		end
 	end

test/test_low_balance.rb 🔗

@@ -10,9 +10,9 @@ CustomerFinancials::REDIS = Minitest::Mock.new
 class LowBalanceTest < Minitest::Test
 	def test_for_locked
 		ExpiringLock::REDIS.expect(
-			:exists,
-			EMPromise.resolve(1),
-			["jmp_customer_low_balance-test"]
+			:set,
+			EMPromise.resolve(nil),
+			["jmp_customer_low_balance-test", Time, "EX", 604800, "NX"]
 		)
 		assert_kind_of LowBalance::Locked, LowBalance.for(customer).sync
 	end
@@ -20,20 +20,15 @@ class LowBalanceTest < Minitest::Test
 
 	def test_for_no_auto_top_up
 		ExpiringLock::REDIS.expect(
-			:exists,
-			EMPromise.resolve(0),
-			["jmp_customer_low_balance-test"]
+			:set,
+			EMPromise.resolve("OK"),
+			["jmp_customer_low_balance-test", Time, "EX", 604800, "NX"]
 		)
 		CustomerFinancials::REDIS.expect(
 			:smembers,
 			EMPromise.resolve([]),
 			["jmp_customer_btc_addresses-test"]
 		)
-		ExpiringLock::REDIS.expect(
-			:setex,
-			EMPromise.resolve(nil),
-			["jmp_customer_low_balance-test", 60 * 60 * 24 * 7, Time]
-		)
 		assert_kind_of(
 			LowBalance,
 			LowBalance.for(customer).sync
@@ -44,14 +39,9 @@ class LowBalanceTest < Minitest::Test
 
 	def test_for_auto_top_up
 		ExpiringLock::REDIS.expect(
-			:exists,
-			EMPromise.resolve(0),
-			["jmp_customer_low_balance-test"]
-		)
-		ExpiringLock::REDIS.expect(
-			:setex,
-			EMPromise.resolve(nil),
-			["jmp_customer_low_balance-test", 60 * 60 * 24 * 7, Time]
+			:set,
+			EMPromise.resolve("OK"),
+			["jmp_customer_low_balance-test", Time, "EX", 604800, "NX"]
 		)
 		assert_kind_of(
 			LowBalance::AutoTopUp,

test/test_web.rb 🔗

@@ -125,9 +125,9 @@ class WebTest < Minitest::Test
 
 	def test_outbound_low_balance
 		ExpiringLock::REDIS.expect(
-			:exists,
-			EMPromise.resolve(1),
-			["jmp_customer_low_balance-customerid_low"]
+			:set,
+			EMPromise.resolve(nil),
+			["jmp_customer_low_balance-customerid_low", Time, "EX", 604800, "NX"]
 		)
 
 		post(
@@ -161,15 +161,9 @@ class WebTest < Minitest::Test
 		)
 
 		ExpiringLock::REDIS.expect(
-			:exists,
-			nil,
-			["jmp_customer_low_balance-customerid_topup"]
-		)
-
-		ExpiringLock::REDIS.expect(
-			:setex,
-			nil,
-			["jmp_customer_low_balance-customerid_topup", Integer, Time]
+			:set,
+			EMPromise.resolve("OK"),
+			["jmp_customer_low_balance-customerid_topup", Time, "EX", 604800, "NX"]
 		)
 
 		Customer::BLATHER.expect(
@@ -307,9 +301,9 @@ class WebTest < Minitest::Test
 
 	def test_inbound_low
 		ExpiringLock::REDIS.expect(
-			:exists,
-			EMPromise.resolve(1),
-			["jmp_customer_low_balance-customerid_low"]
+			:set,
+			EMPromise.resolve(nil),
+			["jmp_customer_low_balance-customerid_low", Time, "EX", 604800, "NX"]
 		)
 
 		post(