2024-01-15 15:43:12 +00:00
|
|
|
type LimitedMapRecord<V> = {
|
|
|
|
value: V;
|
|
|
|
la: number;
|
|
|
|
expTimer?: number;
|
|
|
|
};
|
2023-10-03 12:16:33 +00:00
|
|
|
|
|
|
|
export class LimitedMap<V> {
|
2024-01-13 16:30:15 +00:00
|
|
|
private map: Map<string, LimitedMapRecord<V>>;
|
|
|
|
|
|
|
|
constructor(
|
|
|
|
private maxSize: number,
|
|
|
|
initialJson: Record<string, LimitedMapRecord<V>> = {},
|
|
|
|
) {
|
|
|
|
this.map = new Map(Object.entries(initialJson));
|
2023-10-03 12:16:33 +00:00
|
|
|
}
|
|
|
|
|
2024-01-13 16:30:15 +00:00
|
|
|
/**
|
|
|
|
* @param key
|
|
|
|
* @param value
|
|
|
|
* @param ttl time to live (in ms)
|
|
|
|
*/
|
|
|
|
set(key: string, value: V, ttl?: number) {
|
2024-01-15 15:43:12 +00:00
|
|
|
const entry: LimitedMapRecord<V> = { value, la: Date.now() };
|
2024-01-13 16:30:15 +00:00
|
|
|
if (ttl) {
|
2024-01-15 15:43:12 +00:00
|
|
|
const existingEntry = this.map.get(key);
|
|
|
|
if (existingEntry?.expTimer) {
|
|
|
|
clearTimeout(existingEntry.expTimer);
|
|
|
|
}
|
|
|
|
entry.expTimer = setTimeout(() => {
|
2024-01-13 16:30:15 +00:00
|
|
|
this.map.delete(key);
|
|
|
|
}, ttl);
|
|
|
|
}
|
|
|
|
if (this.map.size >= this.maxSize) {
|
2023-10-03 12:16:33 +00:00
|
|
|
// Remove the oldest key before adding a new one
|
|
|
|
const oldestKey = this.getOldestKey();
|
2024-01-13 16:30:15 +00:00
|
|
|
this.map.delete(oldestKey!);
|
2023-10-03 12:16:33 +00:00
|
|
|
}
|
2024-01-15 15:43:12 +00:00
|
|
|
this.map.set(key, entry);
|
2023-10-03 12:16:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
get(key: string): V | undefined {
|
2024-01-13 16:30:15 +00:00
|
|
|
const entry = this.map.get(key);
|
2023-10-03 12:16:33 +00:00
|
|
|
if (entry) {
|
|
|
|
// Update the last accessed timestamp
|
|
|
|
entry.la = Date.now();
|
|
|
|
return entry.value;
|
|
|
|
}
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
remove(key: string) {
|
2024-01-13 16:30:15 +00:00
|
|
|
this.map.delete(key);
|
2023-10-03 12:16:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
toJSON() {
|
2024-01-13 16:30:15 +00:00
|
|
|
return Object.fromEntries(this.map.entries());
|
2023-10-03 12:16:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private getOldestKey(): string | undefined {
|
|
|
|
let oldestKey: string | undefined;
|
|
|
|
let oldestTimestamp: number | undefined;
|
|
|
|
|
2024-01-13 16:30:15 +00:00
|
|
|
for (const [key, entry] of this.map.entries()) {
|
|
|
|
if (!oldestTimestamp || entry.la < oldestTimestamp) {
|
|
|
|
oldestKey = key;
|
|
|
|
oldestTimestamp = entry.la;
|
2023-10-03 12:16:33 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return oldestKey;
|
|
|
|
}
|
|
|
|
}
|