| 
									
										
										
										
											2020-07-15 17:14:28 +08:00
										 |  |  | /* | 
					
						
							|  |  |  | 	MIT License http://www.opensource.org/licenses/mit-license.php
 | 
					
						
							|  |  |  | 	Author Tobias Koppers @sokra | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | "use strict"; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-10 21:00:18 +08:00
										 |  |  | const { forEachBail } = require("enhanced-resolve"); | 
					
						
							| 
									
										
										
										
											2020-07-28 00:09:48 +08:00
										 |  |  | const asyncLib = require("neo-async"); | 
					
						
							| 
									
										
										
										
											2020-07-15 17:14:28 +08:00
										 |  |  | const getLazyHashedEtag = require("./cache/getLazyHashedEtag"); | 
					
						
							| 
									
										
										
										
											2020-08-19 17:25:24 +08:00
										 |  |  | const mergeEtags = require("./cache/mergeEtags"); | 
					
						
							| 
									
										
										
										
											2020-07-15 17:14:28 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | /** @typedef {import("./Cache")} Cache */ | 
					
						
							|  |  |  | /** @typedef {import("./Cache").Etag} Etag */ | 
					
						
							|  |  |  | /** @typedef {import("./cache/getLazyHashedEtag").HashableObject} HashableObject */ | 
					
						
							| 
									
										
										
										
											2021-09-22 18:12:46 +08:00
										 |  |  | /** @typedef {typeof import("./util/Hash")} HashConstructor */ | 
					
						
							| 
									
										
										
										
											2020-07-15 17:14:28 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-19 17:25:24 +08:00
										 |  |  | /** | 
					
						
							|  |  |  |  * @template T | 
					
						
							|  |  |  |  * @callback CallbackCache | 
					
						
							| 
									
										
										
										
											2024-03-18 23:28:40 +08:00
										 |  |  |  * @param {(Error | null)=} err | 
					
						
							|  |  |  |  * @param {(T | null)=} result | 
					
						
							| 
									
										
										
										
											2020-08-19 17:25:24 +08:00
										 |  |  |  * @returns {void} | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * @template T | 
					
						
							|  |  |  |  * @callback CallbackNormalErrorCache | 
					
						
							| 
									
										
										
										
											2021-12-24 20:27:31 +08:00
										 |  |  |  * @param {(Error | null)=} err | 
					
						
							| 
									
										
										
										
											2020-08-19 17:25:24 +08:00
										 |  |  |  * @param {T=} result | 
					
						
							|  |  |  |  * @returns {void} | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-28 00:09:48 +08:00
										 |  |  | class MultiItemCache { | 
					
						
							|  |  |  | 	/** | 
					
						
							|  |  |  | 	 * @param {ItemCacheFacade[]} items item caches | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	constructor(items) { | 
					
						
							|  |  |  | 		this._items = items; | 
					
						
							| 
									
										
										
										
											2025-05-13 21:03:58 +08:00
										 |  |  | 		// @ts-expect-error expected - returns the single ItemCacheFacade when passed an array of length 1
 | 
					
						
							| 
									
										
										
										
											2024-07-31 09:37:24 +08:00
										 |  |  | 		// eslint-disable-next-line no-constructor-return
 | 
					
						
							| 
									
										
										
										
											2025-05-13 21:03:58 +08:00
										 |  |  | 		if (items.length === 1) return /** @type {ItemCacheFacade} */ (items[0]); | 
					
						
							| 
									
										
										
										
											2020-07-28 00:09:48 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/** | 
					
						
							|  |  |  | 	 * @template T | 
					
						
							|  |  |  | 	 * @param {CallbackCache<T>} callback signals when the value is retrieved | 
					
						
							|  |  |  | 	 * @returns {void} | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	get(callback) { | 
					
						
							| 
									
										
										
										
											2022-01-10 21:00:18 +08:00
										 |  |  | 		forEachBail(this._items, (item, callback) => item.get(callback), callback); | 
					
						
							| 
									
										
										
										
											2020-07-28 00:09:48 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/** | 
					
						
							|  |  |  | 	 * @template T | 
					
						
							|  |  |  | 	 * @returns {Promise<T>} promise with the data | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	getPromise() { | 
					
						
							| 
									
										
										
										
											2023-05-25 06:41:32 +08:00
										 |  |  | 		/** | 
					
						
							|  |  |  | 		 * @param {number} i index | 
					
						
							|  |  |  | 		 * @returns {Promise<T>} promise with the data | 
					
						
							|  |  |  | 		 */ | 
					
						
							| 
									
										
										
										
											2025-07-17 00:13:14 +08:00
										 |  |  | 		const next = (i) => | 
					
						
							|  |  |  | 			this._items[i].getPromise().then((result) => { | 
					
						
							| 
									
										
										
										
											2020-07-28 00:09:48 +08:00
										 |  |  | 				if (result !== undefined) return result; | 
					
						
							|  |  |  | 				if (++i < this._items.length) return next(i); | 
					
						
							|  |  |  | 			}); | 
					
						
							|  |  |  | 		return next(0); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/** | 
					
						
							|  |  |  | 	 * @template T | 
					
						
							|  |  |  | 	 * @param {T} data the value to store | 
					
						
							|  |  |  | 	 * @param {CallbackCache<void>} callback signals when the value is stored | 
					
						
							|  |  |  | 	 * @returns {void} | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	store(data, callback) { | 
					
						
							|  |  |  | 		asyncLib.each( | 
					
						
							|  |  |  | 			this._items, | 
					
						
							|  |  |  | 			(item, callback) => item.store(data, callback), | 
					
						
							|  |  |  | 			callback | 
					
						
							|  |  |  | 		); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/** | 
					
						
							|  |  |  | 	 * @template T | 
					
						
							|  |  |  | 	 * @param {T} data the value to store | 
					
						
							|  |  |  | 	 * @returns {Promise<void>} promise signals when the value is stored | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	storePromise(data) { | 
					
						
							| 
									
										
										
										
											2025-07-17 00:13:14 +08:00
										 |  |  | 		return Promise.all(this._items.map((item) => item.storePromise(data))).then( | 
					
						
							| 
									
										
										
										
											2021-05-11 15:31:46 +08:00
										 |  |  | 			() => {} | 
					
						
							|  |  |  | 		); | 
					
						
							| 
									
										
										
										
											2020-07-28 00:09:48 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-15 17:14:28 +08:00
										 |  |  | class ItemCacheFacade { | 
					
						
							|  |  |  | 	/** | 
					
						
							|  |  |  | 	 * @param {Cache} cache the root cache | 
					
						
							|  |  |  | 	 * @param {string} name the child cache item name | 
					
						
							|  |  |  | 	 * @param {Etag | null} etag the etag | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	constructor(cache, name, etag) { | 
					
						
							|  |  |  | 		this._cache = cache; | 
					
						
							|  |  |  | 		this._name = name; | 
					
						
							|  |  |  | 		this._etag = etag; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/** | 
					
						
							|  |  |  | 	 * @template T | 
					
						
							|  |  |  | 	 * @param {CallbackCache<T>} callback signals when the value is retrieved | 
					
						
							|  |  |  | 	 * @returns {void} | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	get(callback) { | 
					
						
							|  |  |  | 		this._cache.get(this._name, this._etag, callback); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/** | 
					
						
							|  |  |  | 	 * @template T | 
					
						
							|  |  |  | 	 * @returns {Promise<T>} promise with the data | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	getPromise() { | 
					
						
							|  |  |  | 		return new Promise((resolve, reject) => { | 
					
						
							|  |  |  | 			this._cache.get(this._name, this._etag, (err, data) => { | 
					
						
							|  |  |  | 				if (err) { | 
					
						
							|  |  |  | 					reject(err); | 
					
						
							|  |  |  | 				} else { | 
					
						
							|  |  |  | 					resolve(data); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			}); | 
					
						
							|  |  |  | 		}); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/** | 
					
						
							|  |  |  | 	 * @template T | 
					
						
							|  |  |  | 	 * @param {T} data the value to store | 
					
						
							|  |  |  | 	 * @param {CallbackCache<void>} callback signals when the value is stored | 
					
						
							|  |  |  | 	 * @returns {void} | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	store(data, callback) { | 
					
						
							|  |  |  | 		this._cache.store(this._name, this._etag, data, callback); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/** | 
					
						
							|  |  |  | 	 * @template T | 
					
						
							|  |  |  | 	 * @param {T} data the value to store | 
					
						
							|  |  |  | 	 * @returns {Promise<void>} promise signals when the value is stored | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	storePromise(data) { | 
					
						
							|  |  |  | 		return new Promise((resolve, reject) => { | 
					
						
							| 
									
										
										
										
											2025-07-17 00:13:14 +08:00
										 |  |  | 			this._cache.store(this._name, this._etag, data, (err) => { | 
					
						
							| 
									
										
										
										
											2020-07-15 17:14:28 +08:00
										 |  |  | 				if (err) { | 
					
						
							|  |  |  | 					reject(err); | 
					
						
							|  |  |  | 				} else { | 
					
						
							|  |  |  | 					resolve(); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			}); | 
					
						
							|  |  |  | 		}); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-08-19 17:25:24 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/** | 
					
						
							|  |  |  | 	 * @template T | 
					
						
							| 
									
										
										
										
											2025-03-12 09:56:14 +08:00
										 |  |  | 	 * @param {(callback: CallbackNormalErrorCache<T>) => void} computer function to compute the value if not cached | 
					
						
							| 
									
										
										
										
											2020-08-19 17:25:24 +08:00
										 |  |  | 	 * @param {CallbackNormalErrorCache<T>} callback signals when the value is retrieved | 
					
						
							|  |  |  | 	 * @returns {void} | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	provide(computer, callback) { | 
					
						
							|  |  |  | 		this.get((err, cacheEntry) => { | 
					
						
							|  |  |  | 			if (err) return callback(err); | 
					
						
							|  |  |  | 			if (cacheEntry !== undefined) return cacheEntry; | 
					
						
							|  |  |  | 			computer((err, result) => { | 
					
						
							|  |  |  | 				if (err) return callback(err); | 
					
						
							| 
									
										
										
										
											2025-07-17 00:13:14 +08:00
										 |  |  | 				this.store(result, (err) => { | 
					
						
							| 
									
										
										
										
											2020-08-19 17:25:24 +08:00
										 |  |  | 					if (err) return callback(err); | 
					
						
							|  |  |  | 					callback(null, result); | 
					
						
							|  |  |  | 				}); | 
					
						
							|  |  |  | 			}); | 
					
						
							|  |  |  | 		}); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/** | 
					
						
							|  |  |  | 	 * @template T | 
					
						
							| 
									
										
										
										
											2025-03-12 09:56:14 +08:00
										 |  |  | 	 * @param {() => Promise<T> | T} computer function to compute the value if not cached | 
					
						
							| 
									
										
										
										
											2020-08-19 17:25:24 +08:00
										 |  |  | 	 * @returns {Promise<T>} promise with the data | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	async providePromise(computer) { | 
					
						
							|  |  |  | 		const cacheEntry = await this.getPromise(); | 
					
						
							|  |  |  | 		if (cacheEntry !== undefined) return cacheEntry; | 
					
						
							|  |  |  | 		const result = await computer(); | 
					
						
							|  |  |  | 		await this.storePromise(result); | 
					
						
							|  |  |  | 		return result; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-07-15 17:14:28 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class CacheFacade { | 
					
						
							|  |  |  | 	/** | 
					
						
							|  |  |  | 	 * @param {Cache} cache the root cache | 
					
						
							|  |  |  | 	 * @param {string} name the child cache name | 
					
						
							| 
									
										
										
										
											2024-03-18 01:15:44 +08:00
										 |  |  | 	 * @param {(string | HashConstructor)=} hashFunction the hash function to use | 
					
						
							| 
									
										
										
										
											2020-07-15 17:14:28 +08:00
										 |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2021-09-22 18:12:46 +08:00
										 |  |  | 	constructor(cache, name, hashFunction) { | 
					
						
							| 
									
										
										
										
											2020-07-15 17:14:28 +08:00
										 |  |  | 		this._cache = cache; | 
					
						
							|  |  |  | 		this._name = name; | 
					
						
							| 
									
										
										
										
											2021-09-22 18:12:46 +08:00
										 |  |  | 		this._hashFunction = hashFunction; | 
					
						
							| 
									
										
										
										
											2020-07-15 17:14:28 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/** | 
					
						
							|  |  |  | 	 * @param {string} name the child cache name# | 
					
						
							|  |  |  | 	 * @returns {CacheFacade} child cache | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	getChildCache(name) { | 
					
						
							| 
									
										
										
										
											2021-09-22 18:12:46 +08:00
										 |  |  | 		return new CacheFacade( | 
					
						
							|  |  |  | 			this._cache, | 
					
						
							|  |  |  | 			`${this._name}|${name}`, | 
					
						
							|  |  |  | 			this._hashFunction | 
					
						
							|  |  |  | 		); | 
					
						
							| 
									
										
										
										
											2020-07-15 17:14:28 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/** | 
					
						
							|  |  |  | 	 * @param {string} identifier the cache identifier | 
					
						
							|  |  |  | 	 * @param {Etag | null} etag the etag | 
					
						
							|  |  |  | 	 * @returns {ItemCacheFacade} item cache | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	getItemCache(identifier, etag) { | 
					
						
							|  |  |  | 		return new ItemCacheFacade( | 
					
						
							|  |  |  | 			this._cache, | 
					
						
							|  |  |  | 			`${this._name}|${identifier}`, | 
					
						
							|  |  |  | 			etag | 
					
						
							|  |  |  | 		); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/** | 
					
						
							|  |  |  | 	 * @param {HashableObject} obj an hashable object | 
					
						
							|  |  |  | 	 * @returns {Etag} an etag that is lazy hashed | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	getLazyHashedEtag(obj) { | 
					
						
							| 
									
										
										
										
											2021-09-22 18:12:46 +08:00
										 |  |  | 		return getLazyHashedEtag(obj, this._hashFunction); | 
					
						
							| 
									
										
										
										
											2020-07-15 17:14:28 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-19 17:25:24 +08:00
										 |  |  | 	/** | 
					
						
							|  |  |  | 	 * @param {Etag} a an etag | 
					
						
							|  |  |  | 	 * @param {Etag} b another etag | 
					
						
							|  |  |  | 	 * @returns {Etag} an etag that represents both | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	mergeEtags(a, b) { | 
					
						
							|  |  |  | 		return mergeEtags(a, b); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-15 17:14:28 +08:00
										 |  |  | 	/** | 
					
						
							|  |  |  | 	 * @template T | 
					
						
							|  |  |  | 	 * @param {string} identifier the cache identifier | 
					
						
							|  |  |  | 	 * @param {Etag | null} etag the etag | 
					
						
							|  |  |  | 	 * @param {CallbackCache<T>} callback signals when the value is retrieved | 
					
						
							|  |  |  | 	 * @returns {void} | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	get(identifier, etag, callback) { | 
					
						
							|  |  |  | 		this._cache.get(`${this._name}|${identifier}`, etag, callback); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/** | 
					
						
							|  |  |  | 	 * @template T | 
					
						
							|  |  |  | 	 * @param {string} identifier the cache identifier | 
					
						
							|  |  |  | 	 * @param {Etag | null} etag the etag | 
					
						
							|  |  |  | 	 * @returns {Promise<T>} promise with the data | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	getPromise(identifier, etag) { | 
					
						
							|  |  |  | 		return new Promise((resolve, reject) => { | 
					
						
							|  |  |  | 			this._cache.get(`${this._name}|${identifier}`, etag, (err, data) => { | 
					
						
							|  |  |  | 				if (err) { | 
					
						
							|  |  |  | 					reject(err); | 
					
						
							|  |  |  | 				} else { | 
					
						
							|  |  |  | 					resolve(data); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			}); | 
					
						
							|  |  |  | 		}); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/** | 
					
						
							|  |  |  | 	 * @template T | 
					
						
							|  |  |  | 	 * @param {string} identifier the cache identifier | 
					
						
							|  |  |  | 	 * @param {Etag | null} etag the etag | 
					
						
							|  |  |  | 	 * @param {T} data the value to store | 
					
						
							|  |  |  | 	 * @param {CallbackCache<void>} callback signals when the value is stored | 
					
						
							|  |  |  | 	 * @returns {void} | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	store(identifier, etag, data, callback) { | 
					
						
							|  |  |  | 		this._cache.store(`${this._name}|${identifier}`, etag, data, callback); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/** | 
					
						
							|  |  |  | 	 * @template T | 
					
						
							|  |  |  | 	 * @param {string} identifier the cache identifier | 
					
						
							|  |  |  | 	 * @param {Etag | null} etag the etag | 
					
						
							|  |  |  | 	 * @param {T} data the value to store | 
					
						
							|  |  |  | 	 * @returns {Promise<void>} promise signals when the value is stored | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	storePromise(identifier, etag, data) { | 
					
						
							|  |  |  | 		return new Promise((resolve, reject) => { | 
					
						
							| 
									
										
										
										
											2025-07-17 00:13:14 +08:00
										 |  |  | 			this._cache.store(`${this._name}|${identifier}`, etag, data, (err) => { | 
					
						
							| 
									
										
										
										
											2020-07-15 17:14:28 +08:00
										 |  |  | 				if (err) { | 
					
						
							|  |  |  | 					reject(err); | 
					
						
							|  |  |  | 				} else { | 
					
						
							|  |  |  | 					resolve(); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			}); | 
					
						
							|  |  |  | 		}); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-08-19 17:25:24 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/** | 
					
						
							|  |  |  | 	 * @template T | 
					
						
							|  |  |  | 	 * @param {string} identifier the cache identifier | 
					
						
							|  |  |  | 	 * @param {Etag | null} etag the etag | 
					
						
							| 
									
										
										
										
											2025-03-12 09:56:14 +08:00
										 |  |  | 	 * @param {(callback: CallbackNormalErrorCache<T>) => void} computer function to compute the value if not cached | 
					
						
							| 
									
										
										
										
											2020-08-19 17:25:24 +08:00
										 |  |  | 	 * @param {CallbackNormalErrorCache<T>} callback signals when the value is retrieved | 
					
						
							|  |  |  | 	 * @returns {void} | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	provide(identifier, etag, computer, callback) { | 
					
						
							|  |  |  | 		this.get(identifier, etag, (err, cacheEntry) => { | 
					
						
							|  |  |  | 			if (err) return callback(err); | 
					
						
							|  |  |  | 			if (cacheEntry !== undefined) return cacheEntry; | 
					
						
							|  |  |  | 			computer((err, result) => { | 
					
						
							|  |  |  | 				if (err) return callback(err); | 
					
						
							| 
									
										
										
										
											2025-07-17 00:13:14 +08:00
										 |  |  | 				this.store(identifier, etag, result, (err) => { | 
					
						
							| 
									
										
										
										
											2020-08-19 17:25:24 +08:00
										 |  |  | 					if (err) return callback(err); | 
					
						
							|  |  |  | 					callback(null, result); | 
					
						
							|  |  |  | 				}); | 
					
						
							|  |  |  | 			}); | 
					
						
							|  |  |  | 		}); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/** | 
					
						
							|  |  |  | 	 * @template T | 
					
						
							|  |  |  | 	 * @param {string} identifier the cache identifier | 
					
						
							|  |  |  | 	 * @param {Etag | null} etag the etag | 
					
						
							| 
									
										
										
										
											2025-03-12 09:56:14 +08:00
										 |  |  | 	 * @param {() => Promise<T> | T} computer function to compute the value if not cached | 
					
						
							| 
									
										
										
										
											2020-08-19 17:25:24 +08:00
										 |  |  | 	 * @returns {Promise<T>} promise with the data | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	async providePromise(identifier, etag, computer) { | 
					
						
							|  |  |  | 		const cacheEntry = await this.getPromise(identifier, etag); | 
					
						
							|  |  |  | 		if (cacheEntry !== undefined) return cacheEntry; | 
					
						
							|  |  |  | 		const result = await computer(); | 
					
						
							|  |  |  | 		await this.storePromise(identifier, etag, result); | 
					
						
							|  |  |  | 		return result; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-07-15 17:14:28 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | module.exports = CacheFacade; | 
					
						
							|  |  |  | module.exports.ItemCacheFacade = ItemCacheFacade; | 
					
						
							| 
									
										
										
										
											2020-07-28 00:09:48 +08:00
										 |  |  | module.exports.MultiItemCache = MultiItemCache; |