2025-01-10 00:17:29 +00:00
|
|
|
import type { MutexOptions } from "./types";
|
|
|
|
import redis from "redis";
|
|
|
|
|
|
|
|
type RedisClient = redis.RedisClientType<
|
|
|
|
redis.RedisDefaultModules & redis.RedisModules,
|
|
|
|
redis.RedisFunctions,
|
|
|
|
redis.RedisScripts
|
|
|
|
>;
|
|
|
|
|
|
|
|
export class MutexLock {
|
|
|
|
redisClient: RedisClient;
|
2025-01-10 00:19:55 +00:00
|
|
|
mutexOptions: MutexOptions | undefined;
|
2025-01-10 00:17:29 +00:00
|
|
|
|
2025-01-10 00:19:55 +00:00
|
|
|
constructor(redisClient: RedisClient, options?: MutexOptions) {
|
2025-01-10 00:17:29 +00:00
|
|
|
this.mutexOptions = options;
|
|
|
|
this.redisClient = redisClient;
|
|
|
|
}
|
|
|
|
|
|
|
|
static async create(options: MutexOptions) {
|
|
|
|
const redisClient = await redis
|
|
|
|
.createClient({
|
2025-01-10 00:19:55 +00:00
|
|
|
url: `redis://${options?.redis?.host ?? "127.0.0.1"}:${options?.redis?.port ?? 6379}`,
|
2025-01-10 00:17:29 +00:00
|
|
|
})
|
|
|
|
.connect();
|
|
|
|
|
|
|
|
return new MutexLock(redisClient, options);
|
|
|
|
}
|
|
|
|
|
|
|
|
async obtainLock(lockName: string) {
|
|
|
|
const lockIdentifier = `mutexlock:${lockName}`;
|
|
|
|
const releaseFunc = async () => {
|
|
|
|
await this.redisClient.del(lockIdentifier);
|
|
|
|
};
|
|
|
|
|
|
|
|
while (true) {
|
|
|
|
const acquired = await this.redisClient.set(lockIdentifier, "1", {
|
|
|
|
NX: true
|
|
|
|
});
|
|
|
|
|
|
|
|
if (acquired) {
|
|
|
|
await this.redisClient.expire(
|
|
|
|
lockIdentifier,
|
2025-01-10 00:19:55 +00:00
|
|
|
this.mutexOptions?.mutex?.ttl || 60
|
2025-01-10 00:17:29 +00:00
|
|
|
);
|
|
|
|
return releaseFunc;
|
|
|
|
}
|
|
|
|
await new Promise(resolve =>
|
2025-01-10 00:19:55 +00:00
|
|
|
setTimeout(resolve, this.mutexOptions?.mutex?.checkInterval || 100)
|
2025-01-10 00:17:29 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}}
|