You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
	
	
		
			350 lines
		
	
	
		
			8.5 KiB
		
	
	
	
		
			JavaScript
		
	
		
		
			
		
	
	
			350 lines
		
	
	
		
			8.5 KiB
		
	
	
	
		
			JavaScript
		
	
| 
											9 months ago
										 | /* | ||
|  | 	MIT License http://www.opensource.org/licenses/mit-license.php
 | ||
|  | 	Author Tobias Koppers @sokra | ||
|  | */ | ||
|  | 
 | ||
|  | "use strict"; | ||
|  | 
 | ||
|  | const { forEachBail } = require("enhanced-resolve"); | ||
|  | const asyncLib = require("neo-async"); | ||
|  | const getLazyHashedEtag = require("./cache/getLazyHashedEtag"); | ||
|  | const mergeEtags = require("./cache/mergeEtags"); | ||
|  | 
 | ||
|  | /** @typedef {import("./Cache")} Cache */ | ||
|  | /** @typedef {import("./Cache").Etag} Etag */ | ||
|  | /** @typedef {import("./WebpackError")} WebpackError */ | ||
|  | /** @typedef {import("./cache/getLazyHashedEtag").HashableObject} HashableObject */ | ||
|  | /** @typedef {typeof import("./util/Hash")} HashConstructor */ | ||
|  | 
 | ||
|  | /** | ||
|  |  * @template T | ||
|  |  * @callback CallbackCache | ||
|  |  * @param {(WebpackError | null)=} err | ||
|  |  * @param {T=} result | ||
|  |  * @returns {void} | ||
|  |  */ | ||
|  | 
 | ||
|  | /** | ||
|  |  * @template T | ||
|  |  * @callback CallbackNormalErrorCache | ||
|  |  * @param {(Error | null)=} err | ||
|  |  * @param {T=} result | ||
|  |  * @returns {void} | ||
|  |  */ | ||
|  | 
 | ||
|  | class MultiItemCache { | ||
|  | 	/** | ||
|  | 	 * @param {ItemCacheFacade[]} items item caches | ||
|  | 	 */ | ||
|  | 	constructor(items) { | ||
|  | 		this._items = items; | ||
|  | 		if (items.length === 1) return /** @type {any} */ (items[0]); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/** | ||
|  | 	 * @template T | ||
|  | 	 * @param {CallbackCache<T>} callback signals when the value is retrieved | ||
|  | 	 * @returns {void} | ||
|  | 	 */ | ||
|  | 	get(callback) { | ||
|  | 		forEachBail(this._items, (item, callback) => item.get(callback), callback); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/** | ||
|  | 	 * @template T | ||
|  | 	 * @returns {Promise<T>} promise with the data | ||
|  | 	 */ | ||
|  | 	getPromise() { | ||
|  | 		/** | ||
|  | 		 * @param {number} i index | ||
|  | 		 * @returns {Promise<T>} promise with the data | ||
|  | 		 */ | ||
|  | 		const next = i => { | ||
|  | 			return this._items[i].getPromise().then(result => { | ||
|  | 				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) { | ||
|  | 		return Promise.all(this._items.map(item => item.storePromise(data))).then( | ||
|  | 			() => {} | ||
|  | 		); | ||
|  | 	} | ||
|  | } | ||
|  | 
 | ||
|  | 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) => { | ||
|  | 			this._cache.store(this._name, this._etag, data, err => { | ||
|  | 				if (err) { | ||
|  | 					reject(err); | ||
|  | 				} else { | ||
|  | 					resolve(); | ||
|  | 				} | ||
|  | 			}); | ||
|  | 		}); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/** | ||
|  | 	 * @template T | ||
|  | 	 * @param {function(CallbackNormalErrorCache<T>): void} computer function to compute the value if not cached | ||
|  | 	 * @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); | ||
|  | 				this.store(result, err => { | ||
|  | 					if (err) return callback(err); | ||
|  | 					callback(null, result); | ||
|  | 				}); | ||
|  | 			}); | ||
|  | 		}); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/** | ||
|  | 	 * @template T | ||
|  | 	 * @param {function(): Promise<T> | T} computer function to compute the value if not cached | ||
|  | 	 * @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; | ||
|  | 	} | ||
|  | } | ||
|  | 
 | ||
|  | class CacheFacade { | ||
|  | 	/** | ||
|  | 	 * @param {Cache} cache the root cache | ||
|  | 	 * @param {string} name the child cache name | ||
|  | 	 * @param {string | HashConstructor} hashFunction the hash function to use | ||
|  | 	 */ | ||
|  | 	constructor(cache, name, hashFunction) { | ||
|  | 		this._cache = cache; | ||
|  | 		this._name = name; | ||
|  | 		this._hashFunction = hashFunction; | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/** | ||
|  | 	 * @param {string} name the child cache name# | ||
|  | 	 * @returns {CacheFacade} child cache | ||
|  | 	 */ | ||
|  | 	getChildCache(name) { | ||
|  | 		return new CacheFacade( | ||
|  | 			this._cache, | ||
|  | 			`${this._name}|${name}`, | ||
|  | 			this._hashFunction | ||
|  | 		); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/** | ||
|  | 	 * @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) { | ||
|  | 		return getLazyHashedEtag(obj, this._hashFunction); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/** | ||
|  | 	 * @param {Etag} a an etag | ||
|  | 	 * @param {Etag} b another etag | ||
|  | 	 * @returns {Etag} an etag that represents both | ||
|  | 	 */ | ||
|  | 	mergeEtags(a, b) { | ||
|  | 		return mergeEtags(a, b); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/** | ||
|  | 	 * @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) => { | ||
|  | 			this._cache.store(`${this._name}|${identifier}`, etag, data, err => { | ||
|  | 				if (err) { | ||
|  | 					reject(err); | ||
|  | 				} else { | ||
|  | 					resolve(); | ||
|  | 				} | ||
|  | 			}); | ||
|  | 		}); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/** | ||
|  | 	 * @template T | ||
|  | 	 * @param {string} identifier the cache identifier | ||
|  | 	 * @param {Etag | null} etag the etag | ||
|  | 	 * @param {function(CallbackNormalErrorCache<T>): void} computer function to compute the value if not cached | ||
|  | 	 * @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); | ||
|  | 				this.store(identifier, etag, result, err => { | ||
|  | 					if (err) return callback(err); | ||
|  | 					callback(null, result); | ||
|  | 				}); | ||
|  | 			}); | ||
|  | 		}); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/** | ||
|  | 	 * @template T | ||
|  | 	 * @param {string} identifier the cache identifier | ||
|  | 	 * @param {Etag | null} etag the etag | ||
|  | 	 * @param {function(): Promise<T> | T} computer function to compute the value if not cached | ||
|  | 	 * @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; | ||
|  | 	} | ||
|  | } | ||
|  | 
 | ||
|  | module.exports = CacheFacade; | ||
|  | module.exports.ItemCacheFacade = ItemCacheFacade; | ||
|  | module.exports.MultiItemCache = MultiItemCache; |