hippofish/packages/backend/src/misc/cache.ts

121 lines
3 KiB
TypeScript
Raw Normal View History

2023-07-03 02:37:46 +02:00
import { redisClient } from "@/db/redis.js";
import { nativeRandomStr } from "native-utils/built/index.js";
import { encode, decode } from "@msgpack/msgpack";
import { ChainableCommander } from "ioredis";
2021-03-18 02:49:14 +01:00
export class Cache<T> {
2023-07-03 02:37:46 +02:00
private ttl: number;
private fingerprint: string;
constructor(ttl: number) {
this.ttl = ttl;
this.fingerprint = `cache:${nativeRandomStr(32)}`;
}
2021-03-18 02:49:14 +01:00
2023-07-03 02:37:46 +02:00
private prefixedKey(key: string | null): string {
return key ? `${this.fingerprint}:${key}` : this.fingerprint;
}
public async set(key: string | null, value: T, transaction?: ChainableCommander): Promise<void> {
const _key = this.prefixedKey(key);
const _value = Buffer.from(encode(value));
const commander = transaction ?? redisClient;
if (this.ttl === Infinity) {
await commander.set(_key, _value);
} else {
await commander.set(_key, _value, "PX", this.ttl);
}
2021-03-18 02:49:14 +01:00
}
2023-07-03 02:37:46 +02:00
public async get(key: string | null): Promise<T | undefined> {
const _key = this.prefixedKey(key);
const cached = await redisClient.getBuffer(_key);
if (cached === null) return undefined;
return decode(cached) as T;
2021-03-18 02:49:14 +01:00
}
2023-07-03 02:37:46 +02:00
public async getAll(): Promise<Map<string, T>> {
const keys = await redisClient.keys(`${this.fingerprint}*`);
const map = new Map<string, T>();
if (keys.length === 0) {
return map;
}
const values = await redisClient.mgetBuffer(keys);
for (const [i, key] of keys.entries()) {
const val = values[i];
if (val !== null) {
map.set(key, decode(val) as T);
}
2021-03-18 02:49:14 +01:00
}
2023-07-03 02:37:46 +02:00
return map;
2021-03-18 02:49:14 +01:00
}
2023-07-03 02:37:46 +02:00
public async delete(...keys: (string | null)[]): Promise<void> {
if (keys.length > 0) {
const _keys = keys.map(this.prefixedKey);
await redisClient.del(_keys);
}
}
2022-03-20 17:22:00 +01:00
/**
2023-07-03 02:37:46 +02:00
* Returns if cached value exists. Otherwise, calls fetcher and caches.
* Overwrites cached value if invalidated by the optional validator.
2022-03-20 17:22:00 +01:00
*/
2023-01-13 05:40:33 +01:00
public async fetch(
key: string | null,
fetcher: () => Promise<T>,
validator?: (cachedValue: T) => boolean,
): Promise<T> {
2023-07-03 02:37:46 +02:00
const cachedValue = await this.get(key);
if (cachedValue !== undefined) {
2022-03-20 17:22:00 +01:00
if (validator) {
if (validator(cachedValue)) {
// Cache HIT
return cachedValue;
}
} else {
// Cache HIT
return cachedValue;
}
}
// Cache MISS
const value = await fetcher();
2023-07-03 02:37:46 +02:00
await this.set(key, value);
return value;
}
/**
2023-07-03 02:37:46 +02:00
* Returns if cached value exists. Otherwise, calls fetcher and caches if the fetcher returns a value.
* Overwrites cached value if invalidated by the optional validator.
*/
2023-01-13 05:40:33 +01:00
public async fetchMaybe(
key: string | null,
fetcher: () => Promise<T | undefined>,
validator?: (cachedValue: T) => boolean,
): Promise<T | undefined> {
2023-07-03 02:37:46 +02:00
const cachedValue = await this.get(key);
if (cachedValue !== undefined) {
if (validator) {
if (validator(cachedValue)) {
// Cache HIT
return cachedValue;
}
} else {
// Cache HIT
return cachedValue;
}
}
// Cache MISS
const value = await fetcher();
if (value !== undefined) {
2023-07-03 02:37:46 +02:00
await this.set(key, value);
}
return value;
}
2021-03-18 02:49:14 +01:00
}