-
Notifications
You must be signed in to change notification settings - Fork 938
/
Copy pathredislock.go
68 lines (56 loc) · 1.3 KB
/
redislock.go
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
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
package common
import (
"strconv"
"time"
"emperror.dev/errors"
"github.com/mediocregopher/radix/v3"
)
// Locks the lock and if succeded sets it to expire after maxdur
// So that if someting went wrong its not locked forever
func TryLockRedisKey(key string, maxDur int) (bool, error) {
resp := ""
err := RedisPool.Do(radix.Cmd(&resp, "SET", key, "1", "NX", "EX", strconv.Itoa(maxDur)))
if err != nil {
return false, err
}
if resp == "OK" {
return true, nil
}
return false, nil
}
var (
ErrMaxLockAttemptsExceeded = errors.New("Max lock attempts exceeded")
)
// BlockingLockRedisKey blocks until it suceeded to lock the key
func BlockingLockRedisKey(key string, maxTryDuration time.Duration, maxLockDur int) error {
started := time.Now()
sleepDur := time.Millisecond * 100
maxSleep := time.Second
for {
if maxTryDuration != 0 && time.Since(started) > maxTryDuration {
return ErrMaxLockAttemptsExceeded
}
locked, err := TryLockRedisKey(key, maxLockDur)
if err != nil {
return ErrWithCaller(err)
}
if locked {
return nil
}
time.Sleep(sleepDur)
sleepDur *= 2
if sleepDur > maxSleep {
sleepDur = maxSleep
}
}
}
func UnlockRedisKey(key string) {
for {
err := RedisPool.Do(radix.Cmd(nil, "DEL", key))
if err != nil {
time.Sleep(time.Second)
continue
}
break
}
}