-
Notifications
You must be signed in to change notification settings - Fork 2.7k
/
caches.ts
89 lines (87 loc) · 2.76 KB
/
caches.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
import { WeakCache, StrongCache } from "@wry/caches";
interface CleanableCache {
size: number;
max?: number;
clean: () => void;
}
const scheduledCleanup = new WeakSet<CleanableCache>();
function schedule(cache: CleanableCache) {
if (cache.size <= (cache.max || -1)) {
return;
}
if (!scheduledCleanup.has(cache)) {
scheduledCleanup.add(cache);
setTimeout(() => {
cache.clean();
scheduledCleanup.delete(cache);
}, 100);
}
}
/**
* @internal
* A version of WeakCache that will auto-schedule a cleanup of the cache when
* a new item is added and the cache reached maximum size.
* Throttled to once per 100ms.
*
* @privateRemarks
* Should be used throughout the rest of the codebase instead of WeakCache,
* with the notable exception of usage in `wrap` from `optimism` - that one
* already handles cleanup and should remain a `WeakCache`.
*/
export const AutoCleanedWeakCache = function (
max?: number | undefined,
dispose?: ((value: any, key: any) => void) | undefined
) {
/*
Some builds of `WeakCache` are function prototypes, some are classes.
This library still builds with an ES5 target, so we can't extend the
real classes.
Instead, we have to use this workaround until we switch to a newer build
target.
*/
const cache = new WeakCache(max, dispose);
cache.set = function (key: any, value: any) {
const ret = WeakCache.prototype.set.call(this, key, value);
schedule(this as any as CleanableCache);
return ret;
};
return cache;
} as any as typeof WeakCache;
/**
* @internal
*/
export type AutoCleanedWeakCache<K extends object, V> = WeakCache<K, V>;
/**
* @internal
* A version of StrongCache that will auto-schedule a cleanup of the cache when
* a new item is added and the cache reached maximum size.
* Throttled to once per 100ms.
*
* @privateRemarks
* Should be used throughout the rest of the codebase instead of StrongCache,
* with the notable exception of usage in `wrap` from `optimism` - that one
* already handles cleanup and should remain a `StrongCache`.
*/
export const AutoCleanedStrongCache = function (
max?: number | undefined,
dispose?: ((value: any, key: any) => void) | undefined
) {
/*
Some builds of `StrongCache` are function prototypes, some are classes.
This library still builds with an ES5 target, so we can't extend the
real classes.
Instead, we have to use this workaround until we switch to a newer build
target.
*/
const cache = new StrongCache(max, dispose);
cache.set = function (key: any, value: any) {
const ret = StrongCache.prototype.set.call(this, key, value);
schedule(this as any as CleanableCache);
return ret;
};
return cache;
} as any as typeof StrongCache;
/**
* @internal
*/
export type AutoCleanedStrongCache<K, V> = StrongCache<K, V>;