@@ -32,7 +32,6 @@ type consulLock struct {
32
32
isAcquired bool
33
33
isReleased bool
34
34
ttl string
35
- timeout uint64
36
35
sessionID string
37
36
eventChan chan LockEvent
38
37
stopChan chan struct {}
@@ -57,31 +56,15 @@ func (cp *ConsulClient) NewLock(name string, myID string, ttl uint64) (LockInter
57
56
58
57
// Acquire a lock
59
58
func (lk * consulLock ) Acquire (timeout uint64 ) error {
60
- // session configuration
61
- sessCfg := api.SessionEntry {
62
- Name : lk .keyName ,
63
- Behavior : "release" ,
64
- LockDelay : 10 * time .Millisecond ,
65
- TTL : lk .ttl ,
66
- }
67
-
68
59
// Create consul session
69
- sessionID , _ , err := lk .client . Session (). CreateNoChecks ( & sessCfg , nil )
60
+ err := lk .createSession ( )
70
61
if err != nil {
71
62
log .Errorf ("Error Creating session for lock %s. Err: %v" , lk .keyName , err )
72
63
return err
73
64
}
74
65
75
- log .Infof ("Created session: %s for lock %s/%s" , sessionID , lk .name , lk .myID )
76
-
77
- // save the session ID for later
78
- lk .mutex .Lock ()
79
- lk .timeout = timeout
80
- lk .sessionID = sessionID
81
- lk .mutex .Unlock ()
82
-
83
66
// Refresh the session in background
84
- go lk .client . Session (). RenewPeriodic ( lk . ttl , sessionID , nil , lk . stopChan )
67
+ go lk .renewSession ( )
85
68
86
69
// Watch for changes on the lock
87
70
go lk .acquireLock ()
@@ -105,18 +88,18 @@ func (lk *consulLock) Acquire(timeout uint64) error {
105
88
106
89
// Release a lock
107
90
func (lk * consulLock ) Release () error {
108
- lk .mutex .Lock ()
109
- defer lk .mutex .Unlock ()
110
91
111
92
// Mark this as released
93
+ lk .mutex .Lock ()
112
94
lk .isReleased = true
95
+ lk .mutex .Unlock ()
113
96
114
97
// Send stop signal on stop channel
115
98
close (lk .stopChan )
116
99
117
100
// If the lock was acquired, release it
118
- if lk .isAcquired {
119
- lk .isAcquired = false
101
+ if lk .IsAcquired () {
102
+ lk .setAcquired ( false )
120
103
121
104
// Release it via consul client
122
105
succ , _ , err := lk .client .KV ().Release (& api.KVPair {Key : lk .keyName , Value : []byte (lk .myID ), Session : lk .sessionID }, nil )
@@ -164,6 +147,13 @@ func (lk *consulLock) IsAcquired() bool {
164
147
return lk .isAcquired
165
148
}
166
149
150
+ // IsReleased Checks if the lock is released
151
+ func (lk * consulLock ) IsReleased () bool {
152
+ lk .mutex .Lock ()
153
+ defer lk .mutex .Unlock ()
154
+ return lk .isReleased
155
+ }
156
+
167
157
// GetHolder Gets current lock holder's ID
168
158
func (lk * consulLock ) GetHolder () string {
169
159
lk .mutex .Lock ()
@@ -206,25 +196,18 @@ func (lk *consulLock) acquireLock() {
206
196
207
197
log .Debugf ("Got lock(%s) watch Resp: %+v" , lk .myID , resp )
208
198
209
- // check if we are holding the lock
210
- lk .mutex .Lock ()
211
199
// exit the loop if lock is released
212
- if lk .isReleased {
200
+ if lk .IsReleased () {
213
201
log .Infof ("Lock is released. exiting watch" )
214
- lk .mutex .Unlock ()
215
202
return
216
203
}
217
204
218
- isAcquired := lk .isAcquired
219
- lk .mutex .Unlock ()
220
-
221
- if isAcquired {
205
+ // check if we are holding the lock
206
+ if lk .IsAcquired () {
222
207
// check if we lost the lock
223
208
if resp == nil || resp .Session != lk .sessionID || string (resp .Value ) != lk .myID {
224
209
// lock is released
225
- lk .mutex .Lock ()
226
- lk .isAcquired = false
227
- lk .mutex .Unlock ()
210
+ lk .setAcquired (false )
228
211
229
212
log .Infof ("Lost lock %s" , lk .name , lk .myID )
230
213
@@ -248,9 +231,7 @@ func (lk *consulLock) acquireLock() {
248
231
log .Infof ("Acquired lock %s/%s" , lk .name , lk .myID )
249
232
250
233
// Mark the lock as acquired
251
- lk .mutex .Lock ()
252
- lk .isAcquired = true
253
- lk .mutex .Unlock ()
234
+ lk .setAcquired (true )
254
235
255
236
// Send acquired message to event channel
256
237
lk .eventChan <- LockEvent {EventType : LockAcquired }
@@ -263,3 +244,55 @@ func (lk *consulLock) acquireLock() {
263
244
waitIdx = meta .LastIndex
264
245
}
265
246
}
247
+
248
+ // setAcquired marks the lock as acquired/not
249
+ func (lk * consulLock ) setAcquired (isAcquired bool ) {
250
+ lk .mutex .Lock ()
251
+ lk .isAcquired = isAcquired
252
+ lk .mutex .Unlock ()
253
+ }
254
+
255
+ // createSession creates a consul-session for the lock
256
+ func (lk * consulLock ) createSession () error {
257
+ // session configuration
258
+ sessCfg := api.SessionEntry {
259
+ Name : lk .keyName ,
260
+ Behavior : "delete" ,
261
+ LockDelay : 10 * time .Millisecond ,
262
+ TTL : lk .ttl ,
263
+ }
264
+
265
+ // Create consul session
266
+ sessionID , _ , err := lk .client .Session ().CreateNoChecks (& sessCfg , nil )
267
+ if err != nil {
268
+ log .Errorf ("Error Creating session for lock %s. Err: %v" , lk .keyName , err )
269
+ return err
270
+ }
271
+
272
+ log .Infof ("Created session: %s for lock %s/%s" , sessionID , lk .name , lk .myID )
273
+
274
+ // save the session ID for later
275
+ lk .mutex .Lock ()
276
+ lk .sessionID = sessionID
277
+ lk .mutex .Unlock ()
278
+
279
+ return nil
280
+ }
281
+
282
+ // renewSession keeps the session alive.. If a session expires, it creates new one..
283
+ func (lk * consulLock ) renewSession () {
284
+ for {
285
+ err := lk .client .Session ().RenewPeriodic (lk .ttl , lk .sessionID , nil , lk .stopChan )
286
+ if err == nil || lk .IsReleased () {
287
+ // If lock was released, exit this go routine
288
+ return
289
+ }
290
+
291
+ // Create new consul session
292
+ err = lk .createSession ()
293
+ if err != nil {
294
+ log .Errorf ("Error Creating session for lock %s. Err: %v" , lk .keyName , err )
295
+ }
296
+ }
297
+
298
+ }
0 commit comments