yumechi-no-kuni/packages/backend/src/core/AppLockService.ts

45 lines
1.2 KiB
TypeScript
Raw Normal View History

2022-09-17 13:27:08 -05:00
import { promisify } from 'node:util';
import { Inject, Injectable } from '@nestjs/common';
import redisLock from 'redis-lock';
import Redis from 'ioredis';
import { DI } from '@/di-symbols.js';
2022-12-04 02:05:32 -06:00
import { bindThis } from '@/decorators.js';
2022-09-17 13:27:08 -05:00
/**
* Retry delay (ms) for lock acquisition
*/
const retryDelay = 100;
@Injectable()
export class AppLockService {
2023-02-21 23:58:41 -06:00
private lock: (key: string, timeout?: number, _?: (() => Promise<void>) | undefined) => Promise<() => void>;
2022-09-17 13:27:08 -05:00
constructor(
@Inject(DI.redis)
private redisClient: Redis.Redis,
) {
2022-09-18 13:11:50 -05:00
this.lock = promisify(redisLock(this.redisClient, retryDelay));
2022-09-17 13:27:08 -05:00
}
/**
* Get AP Object lock
* @param uri AP object ID
* @param timeout Lock timeout (ms), The timeout releases previous lock.
* @returns Unlock function
*/
@bindThis
2022-09-17 13:27:08 -05:00
public getApLock(uri: string, timeout = 30 * 1000): Promise<() => void> {
2022-09-18 13:11:50 -05:00
return this.lock(`ap-object:${uri}`, timeout);
2022-09-17 13:27:08 -05:00
}
@bindThis
2022-09-17 13:27:08 -05:00
public getFetchInstanceMetadataLock(host: string, timeout = 30 * 1000): Promise<() => void> {
2022-09-18 13:11:50 -05:00
return this.lock(`instance:${host}`, timeout);
2022-09-17 13:27:08 -05:00
}
@bindThis
2022-09-17 13:27:08 -05:00
public getChartInsertLock(lockKey: string, timeout = 30 * 1000): Promise<() => void> {
2022-09-18 13:11:50 -05:00
return this.lock(`chart-insert:${lockKey}`, timeout);
2022-09-17 13:27:08 -05:00
}
}