Add locks to different write operations

This commit is contained in:
Ajay
2023-07-23 23:21:50 -04:00
parent b2081fe155
commit 8bcc781da7
5 changed files with 68 additions and 3 deletions

View File

@@ -1,6 +1,6 @@
import { config } from "../config";
import { Logger } from "./logger";
import { createClient } from "redis";
import { SetOptions, createClient } from "redis";
import { RedisCommandArgument, RedisCommandArguments, RedisCommandRawReply } from "@redis/client/dist/lib/commands";
import { RedisClientOptions } from "@redis/client/dist/lib/client";
import { RedisReply } from "rate-limit-redis";
@@ -16,7 +16,7 @@ export interface RedisStats {
interface RedisSB {
get(key: RedisCommandArgument): Promise<string>;
set(key: RedisCommandArgument, value: RedisCommandArgument): Promise<string>;
set(key: RedisCommandArgument, value: RedisCommandArgument, options?: SetOptions): Promise<string>;
setEx(key: RedisCommandArgument, seconds: number, value: RedisCommandArgument): Promise<string>;
del(...keys: [RedisCommandArgument]): Promise<number>;
increment?(key: RedisCommandArgument): Promise<RedisCommandRawReply[]>;
@@ -125,7 +125,7 @@ if (config.redis?.enabled) {
const set = client.set.bind(client);
const setEx = client.setEx.bind(client);
exportClient.set = (key, value) => setFun(set, [key, value]);
exportClient.set = (key, value, options) => setFun(set, [key, value, options]);
exportClient.setEx = (key, seconds, value) => setFun(setEx, [key, seconds, value]);
exportClient.increment = (key) => new Promise((resolve, reject) =>
void client.multi()

37
src/utils/redisLock.ts Normal file
View File

@@ -0,0 +1,37 @@
import redis from "../utils/redis";
import { Logger } from "./logger";
const defaultTimeout = 5000;
export type AcquiredLock = {
status: false
} | {
status: true;
unlock: () => void;
};
export async function acquireLock(key: string, timeout = defaultTimeout): Promise<AcquiredLock> {
try {
const result = await redis.set(key, "1", {
PX: timeout,
NX: true
});
if (result) {
return {
status: true,
unlock: () => void redis.del(key).catch((err) => Logger.error(err))
};
} else {
return {
status: false
};
}
} catch (e) {
Logger.error(e as string);
}
return {
status: false
};
}