2 unstable releases
Uses new Rust 2024
| 0.2.0 | Dec 11, 2025 |
|---|---|
| 0.1.0 | Nov 19, 2025 |
#2404 in Database interfaces
32KB
530 lines
DLock
A lease based locking implementations for distributed clients with support for a fencing token to prevent usage of stale locks.
Overview
DLock is a lease based distributed lock implementation. It supports recovering from a dead lease holder as well as provides a fencing token to prevent usage of stale tokens.
The implementation has a generic frontend called DLock which accepts a provider backend to support the actual locking implementation. Currently it only provides a AWS DynamoDB provider backend.
Algorithm
The following diagram shows a basic lock contention where Client A has already acquired the lock when Client B attempts to get the lock.
sequenceDiagram
participant A as Client A
participant P as Provider
participant B as Client B
activate A
A ->> P: Acquire (prev_uuid=None, new_uuid=A1, duration=D1)
P ->> A: Acquired (uuid=A1, token=0)
A ->> A: Do synchronized work
activate B
loop
B ->> P: Acquire (prev_uuid=None, new_uuid=B1, duration=D2)
P ->> B: Failure (uuid=A1, duration=D1)
end
A ->> P: Release (uuid=A1)
P ->> A: Released (uuid=A1)
deactivate A
B ->> P: Acquire (prev_uuid=None, new_uuid=B1, duration=D2)
P ->> B: Acquired (uuid=B1, duration=D2, token=0)
B ->> B: Do synchronized work
B ->> P: Release (uuid=B1)
P ->> B: Released (uuid=B1)
deactivate B
The following diagram shows a dead lease recovery scenario where Client A has died after acquiring the lock.
sequenceDiagram
participant A as Client A
participant P as Provider
participant B as Client B
activate A
A ->> P: Acquire (prev_uuid=None, new_uuid=A1, duration=D1)
P ->> A: Acquired (uuid=A1, token=0)
A ->> A: Crash!
deactivate A
activate B
B ->> P: Acquire (prev_uuid=None, new_uuid=B1, duration=D2)
P ->> B: Failure (uuid=A1, duration=D1)
loop D1
B ->> P: Acquire (prev_uuid=None, new_uuid=B1, duration=D2)
P ->> B: Failure (uuid=A1, duration=D1)
end
B ->> P: Acquire (prev_uuid=A1, new_uuid=B1, duration=D2)
P ->> B: Acquired (uuid=B1, duration=D2, token=1)
B ->> B: Synchronized Work
B ->> P: Release (uuid=B1)
P ->> B: Released (uuid=B1)
deactivate B
The following diagram shows a paused lease holder scenario where Client A is interrupted for a long time and the fencing token is used to ensure correct behavior.
sequenceDiagram
participant A as Client A
participant P as Provider
participant S as Storage
participant B as Client B
activate A
A ->> P: Acquire (prev_uuid=None, new_uuid=A1, duration=D1)
P ->> A: Acquired (uuid=A1, token=0)
activate B
B ->> P: Acquire (prev_uuid=None, new_uuid=B1, duration=D2)
P ->> B: Failure (uuid=A1, duration=D1)
loop D1
B ->> P: Acquire (prev_uuid=None, new_uuid=B1, duration=D2)
P ->> B: Failure (uuid=A1, duration=D1)
end
B ->> P: Acquire (prev_uuid=A1, new_uuid=B1, duration=D2)
P ->> B: Acquired (uuid=B1, duration=D2, token=1)
B ->> S: Synchronized Work (token=1)
S ->> B: Success (token=1)
B ->> P: Release (uuid=B1)
P ->> B: Released (uuid=B1)
deactivate B
A ->> S: Synchronized Work (token=0)
S ->> S: Token Invalid 0 < 1
S ->> A: Failure (token=0)
A ->> P: Release (uuid=A1)
P ->> A: Already Released ()
deactivate A
Usage
Add DLock to your crate:
[dependencies]
dlock = "0.1"
Create a provider for the lock backend:
let client = ... // Create a aws_sdk_dynamodb::Client based on your project configuration
let provider = DynamodbProvider::builder()
.client(Arc::new(client))
.table_name("dynamodb_table".to_string())
.build();
You can use the automatic lease renewal mechanism built into DLock or manually manage the lease yourself:
Automatic
with will acquire, automatically renew, and release the lease. For more information see the function documentation.
let lock = DLock::builder()
.name("my_lock".to_string())
.owner("me".to_string())
.duration(Duration::from_secs(1))
.provider(provider)
.build();
let result = lock.with(async |token|
// Synchronized Work
).await?;
Manually
You are responsible for acquiring, renewing, and releasing the lease.
let lock = DLock::builder()
.name("my_lock".to_string())
.owner("me".to_string())
.duration(Duration::from_secs(10))
.provider(provider)
.build();
let lease = lock.acquire().await?;
// Synchronized Work
lease.release().await?;
Dependencies
~8–16MB
~189K SLoC