silverbullet/common/limited_map.ts

64 lines
1.4 KiB
TypeScript
Raw Permalink Normal View History

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