-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathlock.go
68 lines (52 loc) · 1.67 KB
/
lock.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
// mogul - distributed lock via mongoDB
//
// Copyright 2016 - Rene Kroon <[email protected]>
//
package mogul
import (
"time"
"github.com/globalsign/mgo"
"github.com/globalsign/mgo/bson"
)
// Mutex is the object used for locking
type Mutex struct {
doc lock
collection *mgo.Collection
}
type lock struct {
Name string `bson:"_id"`
User string `bson:"user"`
ExpiresAtUtc time.Time `bson:"expires"`
}
// MutexCreator is a specialization of the Manager struct for locking
type MutexCreator interface {
NewMutex(name string, user string) *Mutex
}
// Trylock will claim a lock if it is available. It also returns true when you already hold the lock.
// This extends the duration if you already hold the lock.
func (m *Mutex) TryLock(atMost time.Duration) (bool, error) {
now := time.Now().UTC()
until := now.Add(atMost)
m.doc.ExpiresAtUtc = until
var result lock
selector := bson.M{"$or": []bson.M{m.identityClause(), bson.M{"$and": []bson.M{bson.M{"_id": m.doc.Name}, bson.M{"expires": bson.M{"$lt": now}}}}}}
_, err := m.collection.Find(selector).Apply(mgo.Change{
Update: &m.doc,
ReturnNew: true,
}, &result)
if err == mgo.ErrNotFound {
err = m.collection.Insert(&m.doc)
}
return err == nil, err
}
// Unlock frees the lock, removing the corresponding record in the database
func (m *Mutex) Unlock() error {
return m.collection.Remove(m.identityClause())
}
// IsExpired will return true when your lock time has expired
func (m *Mutex) IsExpired() bool {
return m.doc.ExpiresAtUtc.Before(time.Now().UTC())
}
func (m *Mutex) identityClause() bson.M {
return bson.M{"$and": []bson.M{bson.M{"_id": m.doc.Name}, bson.M{"user": m.doc.User}}}
}