forked from alistairjevans/mongodb-locks
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathMongoDistributedLock.cs
95 lines (88 loc) · 3.83 KB
/
MongoDistributedLock.cs
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
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
using System;
using System.Threading.Tasks;
using MongoDB.Driver;
namespace MongoDistributedLocks
{
public class MongoDistributedLock : IAsyncDisposable
{
private readonly TimeSpan expirationDelay;
private readonly IMongoCollection<LockModel> collection;
private readonly string lockId;
private bool lockAcquired;
public MongoDistributedLock(IMongoCollection<LockModel> collection, string lockId, TimeSpan expirationDelay)
{
this.collection = collection;
this.lockId = lockId;
this.expirationDelay = expirationDelay;
}
/// <summary>
/// Attempts to acquire a lock.
/// </summary>
/// <returns>True if the lock was acquired, false otherwise.</returns>
/// <remarks>
/// The way that a lock is acquired is with a FindAndUpdate operation
/// with an upsert, so that if a lock does not exist, one is created.
/// If the FindAndUpdate output is null, that means the document did not exist,
/// but one has been created. This means the lock now belongs to this thread.
/// If the FindAndModify output is anything else, that means the document exists already,
/// (i.e. the lock has been taken), so this thread must wait.
/// If two threads both attempt to upsert at the same time, one will get a duplicate key exception,
/// meaning it failed to acquire the lock.
/// </remarks>
public async Task<bool> AttemptGetLock()
{
if (lockAcquired)
{
return true;
}
try
{
var response = await collection.FindOneAndUpdateAsync<LockModel>(
// Find a record with the lock ID
x => x.Id == lockId,
// If our 'upsert' creates a document, set the ID to the lock ID and set the expire time.
Builders<LockModel>.Update
.SetOnInsert(x => x.Id, lockId)
.SetOnInsert(x => x.ExpireAt, DateTime.UtcNow + expirationDelay),
new FindOneAndUpdateOptions<LockModel>
{
// If the record doesn't exist, create it.
IsUpsert = true,
// Specifies that the result we want is the record BEFORE it
// was created (this is important).
ReturnDocument = ReturnDocument.Before
}
);
// If the result of the FindOneAndUpdateAsync is null, then that means there was no record
// before we ran our statement, so we now have the lock.
// If the result is not null, then it means a document for the lock already existed, so someone else has the lock.
if (response == null)
{
lockAcquired = true;
}
return lockAcquired;
}
catch (MongoCommandException ex)
{
// 11000 == MongoDB Duplicate Key error
if (ex.Code == 11000)
{
// Two threads have tried to acquire a lock at the exact same moment on the same key,
// which will cause a duplicate key exception in MongoDB.
// So this thread failed to acquire the lock.
return false;
}
// If it's any other error, we rethrow the exception preserving the original stack trace
throw;
}
}
public async ValueTask DisposeAsync()
{
if (!lockAcquired)
{
return;
}
await collection.DeleteOneAsync(x => x.Id == lockId);
}
}
}