| 
									
										
										
										
											2014-07-18 19:31:50 +08:00
										 |  |  | /* | 
					
						
							|  |  |  | 	MIT License http://www.opensource.org/licenses/mit-license.php
 | 
					
						
							|  |  |  | 	Author Tobias Koppers @sokra | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											2018-07-30 23:08:51 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-23 01:00:31 +08:00
										 |  |  | "use strict"; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-26 16:41:57 +08:00
										 |  |  | const NormalModule = require("./NormalModule"); | 
					
						
							| 
									
										
										
										
											2017-11-23 17:59:29 +08:00
										 |  |  | const createHash = require("./util/createHash"); | 
					
						
							| 
									
										
										
										
											2020-12-27 05:32:57 +08:00
										 |  |  | const memoize = require("./util/memoize"); | 
					
						
							| 
									
										
										
										
											2017-11-23 17:59:29 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-22 18:12:46 +08:00
										 |  |  | /** @typedef {import("./ChunkGraph")} ChunkGraph */ | 
					
						
							|  |  |  | /** @typedef {import("./Module")} Module */ | 
					
						
							|  |  |  | /** @typedef {import("./RequestShortener")} RequestShortener */ | 
					
						
							|  |  |  | /** @typedef {typeof import("./util/Hash")} Hash */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-17 19:25:07 +08:00
										 |  |  | /** @typedef {string | RegExp | (string | RegExp)[]} Matcher */ | 
					
						
							| 
									
										
										
										
											2023-04-23 10:43:51 +08:00
										 |  |  | /** @typedef {{test?: Matcher, include?: Matcher, exclude?: Matcher }} MatchObject */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-23 01:00:31 +08:00
										 |  |  | const ModuleFilenameHelpers = exports; | 
					
						
							| 
									
										
										
										
											2014-07-18 19:31:50 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-11 22:09:09 +08:00
										 |  |  | // TODO webpack 6: consider removing these
 | 
					
						
							| 
									
										
										
										
											2014-07-18 19:31:50 +08:00
										 |  |  | ModuleFilenameHelpers.ALL_LOADERS_RESOURCE = "[all-loaders][resource]"; | 
					
						
							| 
									
										
										
										
											2021-05-11 15:31:46 +08:00
										 |  |  | ModuleFilenameHelpers.REGEXP_ALL_LOADERS_RESOURCE = | 
					
						
							|  |  |  | 	/\[all-?loaders\]\[resource\]/gi; | 
					
						
							| 
									
										
										
										
											2014-07-18 19:31:50 +08:00
										 |  |  | ModuleFilenameHelpers.LOADERS_RESOURCE = "[loaders][resource]"; | 
					
						
							|  |  |  | ModuleFilenameHelpers.REGEXP_LOADERS_RESOURCE = /\[loaders\]\[resource\]/gi; | 
					
						
							|  |  |  | ModuleFilenameHelpers.RESOURCE = "[resource]"; | 
					
						
							|  |  |  | ModuleFilenameHelpers.REGEXP_RESOURCE = /\[resource\]/gi; | 
					
						
							|  |  |  | ModuleFilenameHelpers.ABSOLUTE_RESOURCE_PATH = "[absolute-resource-path]"; | 
					
						
							| 
									
										
										
										
											2020-03-13 00:51:26 +08:00
										 |  |  | // cSpell:words olute
 | 
					
						
							| 
									
										
										
										
											2021-05-11 15:31:46 +08:00
										 |  |  | ModuleFilenameHelpers.REGEXP_ABSOLUTE_RESOURCE_PATH = | 
					
						
							|  |  |  | 	/\[abs(olute)?-?resource-?path\]/gi; | 
					
						
							| 
									
										
										
										
											2014-07-18 19:31:50 +08:00
										 |  |  | ModuleFilenameHelpers.RESOURCE_PATH = "[resource-path]"; | 
					
						
							|  |  |  | ModuleFilenameHelpers.REGEXP_RESOURCE_PATH = /\[resource-?path\]/gi; | 
					
						
							|  |  |  | ModuleFilenameHelpers.ALL_LOADERS = "[all-loaders]"; | 
					
						
							|  |  |  | ModuleFilenameHelpers.REGEXP_ALL_LOADERS = /\[all-?loaders\]/gi; | 
					
						
							|  |  |  | ModuleFilenameHelpers.LOADERS = "[loaders]"; | 
					
						
							|  |  |  | ModuleFilenameHelpers.REGEXP_LOADERS = /\[loaders\]/gi; | 
					
						
							|  |  |  | ModuleFilenameHelpers.QUERY = "[query]"; | 
					
						
							|  |  |  | ModuleFilenameHelpers.REGEXP_QUERY = /\[query\]/gi; | 
					
						
							|  |  |  | ModuleFilenameHelpers.ID = "[id]"; | 
					
						
							|  |  |  | ModuleFilenameHelpers.REGEXP_ID = /\[id\]/gi; | 
					
						
							|  |  |  | ModuleFilenameHelpers.HASH = "[hash]"; | 
					
						
							|  |  |  | ModuleFilenameHelpers.REGEXP_HASH = /\[hash\]/gi; | 
					
						
							| 
									
										
										
										
											2017-10-18 06:48:04 +08:00
										 |  |  | ModuleFilenameHelpers.NAMESPACE = "[namespace]"; | 
					
						
							|  |  |  | ModuleFilenameHelpers.REGEXP_NAMESPACE = /\[namespace\]/gi; | 
					
						
							| 
									
										
										
										
											2014-07-18 19:31:50 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-21 20:09:45 +08:00
										 |  |  | /** @typedef {() => string} ReturnStringCallback */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-23 10:43:51 +08:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Returns a function that returns the part of the string after the token | 
					
						
							| 
									
										
										
										
											2024-03-21 20:09:45 +08:00
										 |  |  |  * @param {ReturnStringCallback} strFn the function to get the string | 
					
						
							| 
									
										
										
										
											2023-04-23 10:43:51 +08:00
										 |  |  |  * @param {string} token the token to search for | 
					
						
							| 
									
										
										
										
											2024-03-21 20:09:45 +08:00
										 |  |  |  * @returns {ReturnStringCallback} a function that returns the part of the string after the token | 
					
						
							| 
									
										
										
										
											2023-04-23 10:43:51 +08:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-12-04 18:12:45 +08:00
										 |  |  | const getAfter = (strFn, token) => { | 
					
						
							|  |  |  | 	return () => { | 
					
						
							|  |  |  | 		const str = strFn(); | 
					
						
							|  |  |  | 		const idx = str.indexOf(token); | 
					
						
							| 
									
										
										
										
											2022-03-14 05:54:18 +08:00
										 |  |  | 		return idx < 0 ? "" : str.slice(idx); | 
					
						
							| 
									
										
										
										
											2020-12-04 18:12:45 +08:00
										 |  |  | 	}; | 
					
						
							| 
									
										
										
										
											2017-11-08 18:32:05 +08:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2014-07-18 19:31:50 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-23 10:43:51 +08:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Returns a function that returns the part of the string before the token | 
					
						
							| 
									
										
										
										
											2024-03-21 20:09:45 +08:00
										 |  |  |  * @param {ReturnStringCallback} strFn the function to get the string | 
					
						
							| 
									
										
										
										
											2023-04-23 10:43:51 +08:00
										 |  |  |  * @param {string} token the token to search for | 
					
						
							| 
									
										
										
										
											2024-03-21 20:09:45 +08:00
										 |  |  |  * @returns {ReturnStringCallback} a function that returns the part of the string before the token | 
					
						
							| 
									
										
										
										
											2023-04-23 10:43:51 +08:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-12-04 18:12:45 +08:00
										 |  |  | const getBefore = (strFn, token) => { | 
					
						
							|  |  |  | 	return () => { | 
					
						
							|  |  |  | 		const str = strFn(); | 
					
						
							|  |  |  | 		const idx = str.lastIndexOf(token); | 
					
						
							| 
									
										
										
										
											2022-03-14 05:54:18 +08:00
										 |  |  | 		return idx < 0 ? "" : str.slice(0, idx); | 
					
						
							| 
									
										
										
										
											2020-12-04 18:12:45 +08:00
										 |  |  | 	}; | 
					
						
							| 
									
										
										
										
											2017-11-08 18:32:05 +08:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2014-07-18 19:31:50 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-23 10:43:51 +08:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Returns a function that returns a hash of the string | 
					
						
							| 
									
										
										
										
											2024-03-21 20:09:45 +08:00
										 |  |  |  * @param {ReturnStringCallback} strFn the function to get the string | 
					
						
							|  |  |  |  * @param {string | Hash=} hashFunction the hash function to use | 
					
						
							|  |  |  |  * @returns {ReturnStringCallback} a function that returns the hash of the string | 
					
						
							| 
									
										
										
										
											2023-04-23 10:43:51 +08:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2024-03-21 20:09:45 +08:00
										 |  |  | const getHash = (strFn, hashFunction = "md4") => { | 
					
						
							| 
									
										
										
										
											2020-12-04 18:12:45 +08:00
										 |  |  | 	return () => { | 
					
						
							| 
									
										
										
										
											2021-09-22 18:12:46 +08:00
										 |  |  | 		const hash = createHash(hashFunction); | 
					
						
							| 
									
										
										
										
											2020-12-04 18:12:45 +08:00
										 |  |  | 		hash.update(strFn()); | 
					
						
							|  |  |  | 		const digest = /** @type {string} */ (hash.digest("hex")); | 
					
						
							| 
									
										
										
										
											2022-03-14 05:54:18 +08:00
										 |  |  | 		return digest.slice(0, 4); | 
					
						
							| 
									
										
										
										
											2020-12-04 18:12:45 +08:00
										 |  |  | 	}; | 
					
						
							| 
									
										
										
										
											2017-11-08 18:32:05 +08:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2014-07-18 19:31:50 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-23 10:43:51 +08:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Returns a function that returns the string with the token replaced with the replacement | 
					
						
							|  |  |  |  * @param {string|RegExp} test A regular expression string or Regular Expression object | 
					
						
							|  |  |  |  * @returns	{RegExp} A regular expression object | 
					
						
							|  |  |  |  * @example | 
					
						
							|  |  |  |  * ```js
 | 
					
						
							|  |  |  |  * const test = asRegExp("test"); | 
					
						
							|  |  |  |  * test.test("test"); // true
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * const test2 = asRegExp(/test/); | 
					
						
							|  |  |  |  * test2.test("test"); // true
 | 
					
						
							|  |  |  |  * ```
 | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2017-11-08 18:32:05 +08:00
										 |  |  | const asRegExp = test => { | 
					
						
							| 
									
										
										
										
											2018-05-29 20:50:40 +08:00
										 |  |  | 	if (typeof test === "string") { | 
					
						
							| 
									
										
										
										
											2023-04-23 10:43:51 +08:00
										 |  |  | 		// Escape special characters in the string to prevent them from being interpreted as special characters in a regular expression. Do this by
 | 
					
						
							|  |  |  | 		// adding a backslash before each special character
 | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		test = new RegExp("^" + test.replace(/[-[\]{}()*+?.,\\^$|#\s]/g, "\\$&")); | 
					
						
							| 
									
										
										
										
											2018-05-29 20:50:40 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-03-21 02:00:39 +08:00
										 |  |  | 	return test; | 
					
						
							| 
									
										
										
										
											2017-11-08 18:32:05 +08:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2015-03-21 02:00:39 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-23 10:43:51 +08:00
										 |  |  | /** | 
					
						
							|  |  |  |  * @template T | 
					
						
							|  |  |  |  * Returns a lazy object. The object is lazy in the sense that the properties are | 
					
						
							|  |  |  |  * only evaluated when they are accessed. This is only obtained by setting a function as the value for each key. | 
					
						
							| 
									
										
										
										
											2024-02-20 00:46:07 +08:00
										 |  |  |  * @param {Record<string, () => T>} obj the object to convert to a lazy access object | 
					
						
							| 
									
										
										
										
											2023-04-23 10:43:51 +08:00
										 |  |  |  * @returns {Object} the lazy access object | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-12-04 18:12:45 +08:00
										 |  |  | const lazyObject = obj => { | 
					
						
							|  |  |  | 	const newObj = {}; | 
					
						
							|  |  |  | 	for (const key of Object.keys(obj)) { | 
					
						
							|  |  |  | 		const fn = obj[key]; | 
					
						
							|  |  |  | 		Object.defineProperty(newObj, key, { | 
					
						
							|  |  |  | 			get: () => fn(), | 
					
						
							|  |  |  | 			set: v => { | 
					
						
							|  |  |  | 				Object.defineProperty(newObj, key, { | 
					
						
							|  |  |  | 					value: v, | 
					
						
							|  |  |  | 					enumerable: true, | 
					
						
							|  |  |  | 					writable: true | 
					
						
							|  |  |  | 				}); | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			enumerable: true, | 
					
						
							|  |  |  | 			configurable: true | 
					
						
							|  |  |  | 		}); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return newObj; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-23 10:43:51 +08:00
										 |  |  | const SQUARE_BRACKET_TAG_REGEXP = /\[\\*([\w-]+)\\*\]/gi; | 
					
						
							| 
									
										
										
										
											2020-11-11 22:09:09 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-22 18:12:46 +08:00
										 |  |  | /** | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @param {Module | string} module the module | 
					
						
							|  |  |  |  * @param {TODO} options options | 
					
						
							|  |  |  |  * @param {Object} contextInfo context info | 
					
						
							|  |  |  |  * @param {RequestShortener} contextInfo.requestShortener requestShortener | 
					
						
							|  |  |  |  * @param {ChunkGraph} contextInfo.chunkGraph chunk graph | 
					
						
							| 
									
										
										
										
											2024-03-21 20:09:45 +08:00
										 |  |  |  * @param {string | Hash=} contextInfo.hashFunction the hash function to use | 
					
						
							| 
									
										
										
										
											2021-09-22 18:12:46 +08:00
										 |  |  |  * @returns {string} the filename | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2018-08-28 17:50:33 +08:00
										 |  |  | ModuleFilenameHelpers.createFilename = ( | 
					
						
							| 
									
										
										
										
											2021-09-22 18:12:46 +08:00
										 |  |  | 	module = "", | 
					
						
							| 
									
										
										
										
											2018-08-28 17:50:33 +08:00
										 |  |  | 	options, | 
					
						
							| 
									
										
										
										
											2021-09-22 18:12:46 +08:00
										 |  |  | 	{ requestShortener, chunkGraph, hashFunction = "md4" } | 
					
						
							| 
									
										
										
										
											2018-08-28 17:50:33 +08:00
										 |  |  | ) => { | 
					
						
							| 
									
										
										
										
											2019-06-19 19:16:05 +08:00
										 |  |  | 	const opts = { | 
					
						
							|  |  |  | 		namespace: "", | 
					
						
							|  |  |  | 		moduleFilenameTemplate: "", | 
					
						
							|  |  |  | 		...(typeof options === "object" | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 			? options | 
					
						
							|  |  |  | 			: { | 
					
						
							|  |  |  | 					moduleFilenameTemplate: options | 
					
						
							| 
									
										
										
										
											2024-01-14 09:41:34 +08:00
										 |  |  | 				}) | 
					
						
							| 
									
										
										
										
											2019-06-19 19:16:05 +08:00
										 |  |  | 	}; | 
					
						
							| 
									
										
										
										
											2017-10-19 06:24:59 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-23 01:00:31 +08:00
										 |  |  | 	let absoluteResourcePath; | 
					
						
							|  |  |  | 	let hash; | 
					
						
							| 
									
										
										
										
											2024-03-21 20:09:45 +08:00
										 |  |  | 	/** @type {ReturnStringCallback} */ | 
					
						
							| 
									
										
										
										
											2017-02-23 01:00:31 +08:00
										 |  |  | 	let identifier; | 
					
						
							| 
									
										
										
										
											2024-03-21 20:09:45 +08:00
										 |  |  | 	/** @type {ReturnStringCallback} */ | 
					
						
							| 
									
										
										
										
											2017-02-23 01:00:31 +08:00
										 |  |  | 	let moduleId; | 
					
						
							| 
									
										
										
										
											2024-03-21 20:09:45 +08:00
										 |  |  | 	/** @type {ReturnStringCallback} */ | 
					
						
							| 
									
										
										
										
											2017-02-23 01:00:31 +08:00
										 |  |  | 	let shortIdentifier; | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 	if (typeof module === "string") { | 
					
						
							| 
									
										
										
										
											2024-03-21 20:09:45 +08:00
										 |  |  | 		shortIdentifier = | 
					
						
							|  |  |  | 			/** @type {ReturnStringCallback} */ | 
					
						
							|  |  |  | 			(memoize(() => requestShortener.shorten(module))); | 
					
						
							| 
									
										
										
										
											2016-01-19 08:52:28 +08:00
										 |  |  | 		identifier = shortIdentifier; | 
					
						
							| 
									
										
										
										
											2020-12-04 18:12:45 +08:00
										 |  |  | 		moduleId = () => ""; | 
					
						
							|  |  |  | 		absoluteResourcePath = () => module.split("!").pop(); | 
					
						
							| 
									
										
										
										
											2021-09-22 18:12:46 +08:00
										 |  |  | 		hash = getHash(identifier, hashFunction); | 
					
						
							| 
									
										
										
										
											2014-07-18 19:31:50 +08:00
										 |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2020-12-27 05:32:57 +08:00
										 |  |  | 		shortIdentifier = memoize(() => | 
					
						
							| 
									
										
										
										
											2020-12-04 18:12:45 +08:00
										 |  |  | 			module.readableIdentifier(requestShortener) | 
					
						
							|  |  |  | 		); | 
					
						
							| 
									
										
										
										
											2024-03-21 20:09:45 +08:00
										 |  |  | 		identifier = | 
					
						
							|  |  |  | 			/** @type {ReturnStringCallback} */ | 
					
						
							|  |  |  | 			(memoize(() => requestShortener.shorten(module.identifier()))); | 
					
						
							|  |  |  | 		moduleId = | 
					
						
							|  |  |  | 			/** @type {ReturnStringCallback} */ | 
					
						
							|  |  |  | 			(() => chunkGraph.getModuleId(module)); | 
					
						
							| 
									
										
										
										
											2021-11-16 23:59:28 +08:00
										 |  |  | 		absoluteResourcePath = () => | 
					
						
							| 
									
										
										
										
											2021-11-26 16:41:57 +08:00
										 |  |  | 			module instanceof NormalModule | 
					
						
							|  |  |  | 				? module.resource | 
					
						
							|  |  |  | 				: module.identifier().split("!").pop(); | 
					
						
							| 
									
										
										
										
											2021-09-22 18:12:46 +08:00
										 |  |  | 		hash = getHash(identifier, hashFunction); | 
					
						
							| 
									
										
										
										
											2014-07-18 19:31:50 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-03-21 20:09:45 +08:00
										 |  |  | 	const resource = | 
					
						
							|  |  |  | 		/** @type {ReturnStringCallback} */ | 
					
						
							|  |  |  | 		(memoize(() => shortIdentifier().split("!").pop())); | 
					
						
							| 
									
										
										
										
											2020-11-05 03:23:24 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-23 01:00:31 +08:00
										 |  |  | 	const loaders = getBefore(shortIdentifier, "!"); | 
					
						
							|  |  |  | 	const allLoaders = getBefore(identifier, "!"); | 
					
						
							|  |  |  | 	const query = getAfter(resource, "?"); | 
					
						
							| 
									
										
										
										
											2020-12-04 18:12:45 +08:00
										 |  |  | 	const resourcePath = () => { | 
					
						
							|  |  |  | 		const q = query().length; | 
					
						
							|  |  |  | 		return q === 0 ? resource() : resource().slice(0, -q); | 
					
						
							|  |  |  | 	}; | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 	if (typeof opts.moduleFilenameTemplate === "function") { | 
					
						
							| 
									
										
										
										
											2020-12-04 18:12:45 +08:00
										 |  |  | 		return opts.moduleFilenameTemplate( | 
					
						
							|  |  |  | 			lazyObject({ | 
					
						
							|  |  |  | 				identifier: identifier, | 
					
						
							|  |  |  | 				shortIdentifier: shortIdentifier, | 
					
						
							|  |  |  | 				resource: resource, | 
					
						
							| 
									
										
										
										
											2020-12-27 05:32:57 +08:00
										 |  |  | 				resourcePath: memoize(resourcePath), | 
					
						
							|  |  |  | 				absoluteResourcePath: memoize(absoluteResourcePath), | 
					
						
							| 
									
										
										
										
											2023-02-25 01:38:01 +08:00
										 |  |  | 				loaders: memoize(loaders), | 
					
						
							| 
									
										
										
										
											2020-12-27 05:32:57 +08:00
										 |  |  | 				allLoaders: memoize(allLoaders), | 
					
						
							|  |  |  | 				query: memoize(query), | 
					
						
							|  |  |  | 				moduleId: memoize(moduleId), | 
					
						
							|  |  |  | 				hash: memoize(hash), | 
					
						
							| 
									
										
										
										
											2020-12-04 18:12:45 +08:00
										 |  |  | 				namespace: () => opts.namespace | 
					
						
							|  |  |  | 			}) | 
					
						
							|  |  |  | 		); | 
					
						
							| 
									
										
										
										
											2015-07-07 20:39:48 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-11-11 22:09:09 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// TODO webpack 6: consider removing alternatives without dashes
 | 
					
						
							| 
									
										
										
										
											2020-12-04 18:12:45 +08:00
										 |  |  | 	/** @type {Map<string, function(): string>} */ | 
					
						
							| 
									
										
										
										
											2020-11-11 22:09:09 +08:00
										 |  |  | 	const replacements = new Map([ | 
					
						
							|  |  |  | 		["identifier", identifier], | 
					
						
							|  |  |  | 		["short-identifier", shortIdentifier], | 
					
						
							|  |  |  | 		["resource", resource], | 
					
						
							|  |  |  | 		["resource-path", resourcePath], | 
					
						
							|  |  |  | 		// cSpell:words resourcepath
 | 
					
						
							|  |  |  | 		["resourcepath", resourcePath], | 
					
						
							|  |  |  | 		["absolute-resource-path", absoluteResourcePath], | 
					
						
							|  |  |  | 		["abs-resource-path", absoluteResourcePath], | 
					
						
							|  |  |  | 		// cSpell:words absoluteresource
 | 
					
						
							|  |  |  | 		["absoluteresource-path", absoluteResourcePath], | 
					
						
							|  |  |  | 		// cSpell:words absresource
 | 
					
						
							|  |  |  | 		["absresource-path", absoluteResourcePath], | 
					
						
							|  |  |  | 		// cSpell:words resourcepath
 | 
					
						
							|  |  |  | 		["absolute-resourcepath", absoluteResourcePath], | 
					
						
							|  |  |  | 		// cSpell:words resourcepath
 | 
					
						
							|  |  |  | 		["abs-resourcepath", absoluteResourcePath], | 
					
						
							|  |  |  | 		// cSpell:words absoluteresourcepath
 | 
					
						
							|  |  |  | 		["absoluteresourcepath", absoluteResourcePath], | 
					
						
							|  |  |  | 		// cSpell:words absresourcepath
 | 
					
						
							|  |  |  | 		["absresourcepath", absoluteResourcePath], | 
					
						
							|  |  |  | 		["all-loaders", allLoaders], | 
					
						
							|  |  |  | 		// cSpell:words allloaders
 | 
					
						
							|  |  |  | 		["allloaders", allLoaders], | 
					
						
							|  |  |  | 		["loaders", loaders], | 
					
						
							|  |  |  | 		["query", query], | 
					
						
							|  |  |  | 		["id", moduleId], | 
					
						
							|  |  |  | 		["hash", hash], | 
					
						
							| 
									
										
										
										
											2020-12-04 18:12:45 +08:00
										 |  |  | 		["namespace", () => opts.namespace] | 
					
						
							| 
									
										
										
										
											2020-11-11 22:09:09 +08:00
										 |  |  | 	]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// TODO webpack 6: consider removing weird double placeholders
 | 
					
						
							| 
									
										
										
										
											2024-03-21 20:09:45 +08:00
										 |  |  | 	return /** @type {string} */ (opts.moduleFilenameTemplate) | 
					
						
							| 
									
										
										
										
											2020-11-11 22:09:09 +08:00
										 |  |  | 		.replace(ModuleFilenameHelpers.REGEXP_ALL_LOADERS_RESOURCE, "[identifier]") | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		.replace( | 
					
						
							| 
									
										
										
										
											2020-11-11 22:09:09 +08:00
										 |  |  | 			ModuleFilenameHelpers.REGEXP_LOADERS_RESOURCE, | 
					
						
							|  |  |  | 			"[short-identifier]" | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		) | 
					
						
							| 
									
										
										
										
											2023-04-23 10:43:51 +08:00
										 |  |  | 		.replace(SQUARE_BRACKET_TAG_REGEXP, (match, content) => { | 
					
						
							| 
									
										
										
										
											2020-11-11 22:09:09 +08:00
										 |  |  | 			if (content.length + 2 === match.length) { | 
					
						
							|  |  |  | 				const replacement = replacements.get(content.toLowerCase()); | 
					
						
							|  |  |  | 				if (replacement !== undefined) { | 
					
						
							| 
									
										
										
										
											2020-12-04 18:12:45 +08:00
										 |  |  | 					return replacement(); | 
					
						
							| 
									
										
										
										
											2020-11-11 22:09:09 +08:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} else if (match.startsWith("[\\") && match.endsWith("\\]")) { | 
					
						
							|  |  |  | 				return `[${match.slice(2, -2)}]`; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			return match; | 
					
						
							|  |  |  | 		}); | 
					
						
							| 
									
										
										
										
											2014-07-18 19:31:50 +08:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-23 10:43:51 +08:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Replaces duplicate items in an array with new values generated by a callback function. | 
					
						
							|  |  |  |  * The callback function is called with the duplicate item, the index of the duplicate item, and the number of times the item has been replaced. | 
					
						
							|  |  |  |  * The callback function should return the new value for the duplicate item. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @template T | 
					
						
							|  |  |  |  * @param {T[]} array the array with duplicates to be replaced | 
					
						
							|  |  |  |  * @param {(duplicateItem: T, duplicateItemIndex: number, numberOfTimesReplaced: number) => T} fn callback function to generate new values for the duplicate items | 
					
						
							|  |  |  |  * @param {(firstElement:T, nextElement:T) => -1 | 0 | 1} [comparator] optional comparator function to sort the duplicate items | 
					
						
							|  |  |  |  * @returns {T[]} the array with duplicates replaced | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @example | 
					
						
							|  |  |  |  * ```js
 | 
					
						
							|  |  |  |  * const array = ["a", "b", "c", "a", "b", "a"]; | 
					
						
							|  |  |  |  * const result = ModuleFilenameHelpers.replaceDuplicates(array, (item, index, count) => `${item}-${count}`); | 
					
						
							|  |  |  |  * // result: ["a-1", "b-1", "c", "a-2", "b-2", "a-3"]
 | 
					
						
							|  |  |  |  * ```
 | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2017-11-08 18:32:05 +08:00
										 |  |  | ModuleFilenameHelpers.replaceDuplicates = (array, fn, comparator) => { | 
					
						
							| 
									
										
										
										
											2017-02-23 17:41:42 +08:00
										 |  |  | 	const countMap = Object.create(null); | 
					
						
							|  |  |  | 	const posMap = Object.create(null); | 
					
						
							| 
									
										
										
										
											2023-04-23 10:43:51 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-23 01:00:31 +08:00
										 |  |  | 	array.forEach((item, idx) => { | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		countMap[item] = countMap[item] || []; | 
					
						
							| 
									
										
										
										
											2014-07-26 23:54:00 +08:00
										 |  |  | 		countMap[item].push(idx); | 
					
						
							| 
									
										
										
										
											2014-07-18 19:31:50 +08:00
										 |  |  | 		posMap[item] = 0; | 
					
						
							|  |  |  | 	}); | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 	if (comparator) { | 
					
						
							| 
									
										
										
										
											2017-02-23 01:00:31 +08:00
										 |  |  | 		Object.keys(countMap).forEach(item => { | 
					
						
							| 
									
										
										
										
											2014-07-26 23:54:00 +08:00
										 |  |  | 			countMap[item].sort(comparator); | 
					
						
							|  |  |  | 		}); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-02-23 01:00:31 +08:00
										 |  |  | 	return array.map((item, i) => { | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 		if (countMap[item].length > 1) { | 
					
						
							|  |  |  | 			if (comparator && countMap[item][0] === i) return item; | 
					
						
							| 
									
										
										
										
											2014-07-18 19:31:50 +08:00
										 |  |  | 			return fn(item, i, posMap[item]++); | 
					
						
							| 
									
										
										
										
											2018-05-29 20:50:40 +08:00
										 |  |  | 		} else { | 
					
						
							|  |  |  | 			return item; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2014-07-18 19:31:50 +08:00
										 |  |  | 	}); | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2015-03-21 02:00:39 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-23 10:43:51 +08:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Tests if a string matches a RegExp or an array of RegExp. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @param {string} str string to test | 
					
						
							|  |  |  |  * @param {Matcher} test value which will be used to match against the string | 
					
						
							|  |  |  |  * @returns {boolean} true, when the RegExp matches | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @example | 
					
						
							|  |  |  |  * ```js
 | 
					
						
							|  |  |  |  * ModuleFilenameHelpers.matchPart("foo.js", "foo"); // true
 | 
					
						
							|  |  |  |  * ModuleFilenameHelpers.matchPart("foo.js", "foo.js"); // true
 | 
					
						
							|  |  |  |  * ModuleFilenameHelpers.matchPart("foo.js", "foo."); // false
 | 
					
						
							|  |  |  |  * ModuleFilenameHelpers.matchPart("foo.js", "foo*"); // false
 | 
					
						
							|  |  |  |  * ModuleFilenameHelpers.matchPart("foo.js", "foo.*"); // true
 | 
					
						
							|  |  |  |  * ModuleFilenameHelpers.matchPart("foo.js", /^foo/); // true
 | 
					
						
							|  |  |  |  * ModuleFilenameHelpers.matchPart("foo.js", [/^foo/, "bar"]); // true
 | 
					
						
							|  |  |  |  * ModuleFilenameHelpers.matchPart("foo.js", [/^foo/, "bar"]); // true
 | 
					
						
							|  |  |  |  * ModuleFilenameHelpers.matchPart("foo.js", [/^foo/, /^bar/]); // true
 | 
					
						
							|  |  |  |  * ModuleFilenameHelpers.matchPart("foo.js", [/^baz/, /^bar/]); // false
 | 
					
						
							|  |  |  |  * ```
 | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2017-11-08 18:32:05 +08:00
										 |  |  | ModuleFilenameHelpers.matchPart = (str, test) => { | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 	if (!test) return true; | 
					
						
							| 
									
										
										
										
											2023-04-23 10:43:51 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-25 09:00:20 +08:00
										 |  |  | 	if (Array.isArray(test)) { | 
					
						
							| 
									
										
										
										
											2017-11-08 18:32:05 +08:00
										 |  |  | 		return test.map(asRegExp).some(regExp => regExp.test(str)); | 
					
						
							| 
									
										
										
										
											2015-03-21 02:00:39 +08:00
										 |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2023-04-23 10:43:51 +08:00
										 |  |  | 		return asRegExp(test).test(str); | 
					
						
							| 
									
										
										
										
											2015-03-21 02:00:39 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-23 10:43:51 +08:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Tests if a string matches a match object. The match object can have the following properties: | 
					
						
							|  |  |  |  * - `test`: a RegExp or an array of RegExp | 
					
						
							|  |  |  |  * - `include`: a RegExp or an array of RegExp | 
					
						
							|  |  |  |  * - `exclude`: a RegExp or an array of RegExp | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The `test` property is tested first, then `include` and then `exclude`. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @param {MatchObject} obj a match object to test against the string | 
					
						
							|  |  |  |  * @param {string} str string to test against the matching object | 
					
						
							|  |  |  |  * @returns {boolean} true, when the object matches | 
					
						
							|  |  |  |  * @example | 
					
						
							|  |  |  |  * ```js
 | 
					
						
							|  |  |  |  * ModuleFilenameHelpers.matchObject({ test: "foo.js" }, "foo.js"); // true
 | 
					
						
							|  |  |  |  * ModuleFilenameHelpers.matchObject({ test: /^foo/ }, "foo.js"); // true
 | 
					
						
							|  |  |  |  * ModuleFilenameHelpers.matchObject({ test: [/^foo/, "bar"] }, "foo.js"); // true
 | 
					
						
							|  |  |  |  * ModuleFilenameHelpers.matchObject({ test: [/^foo/, "bar"] }, "baz.js"); // false
 | 
					
						
							|  |  |  |  * ModuleFilenameHelpers.matchObject({ include: "foo.js" }, "foo.js"); // true
 | 
					
						
							|  |  |  |  * ModuleFilenameHelpers.matchObject({ include: "foo.js" }, "bar.js"); // false
 | 
					
						
							|  |  |  |  * ModuleFilenameHelpers.matchObject({ include: /^foo/ }, "foo.js"); // true
 | 
					
						
							|  |  |  |  * ModuleFilenameHelpers.matchObject({ include: [/^foo/, "bar"] }, "foo.js"); // true
 | 
					
						
							|  |  |  |  * ModuleFilenameHelpers.matchObject({ include: [/^foo/, "bar"] }, "baz.js"); // false
 | 
					
						
							|  |  |  |  * ModuleFilenameHelpers.matchObject({ exclude: "foo.js" }, "foo.js"); // false
 | 
					
						
							|  |  |  |  * ModuleFilenameHelpers.matchObject({ exclude: [/^foo/, "bar"] }, "foo.js"); // false
 | 
					
						
							|  |  |  |  * ```
 | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2017-11-08 18:32:05 +08:00
										 |  |  | ModuleFilenameHelpers.matchObject = (obj, str) => { | 
					
						
							| 
									
										
										
										
											2018-05-29 20:50:40 +08:00
										 |  |  | 	if (obj.test) { | 
					
						
							|  |  |  | 		if (!ModuleFilenameHelpers.matchPart(str, obj.test)) { | 
					
						
							|  |  |  | 			return false; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (obj.include) { | 
					
						
							|  |  |  | 		if (!ModuleFilenameHelpers.matchPart(str, obj.include)) { | 
					
						
							|  |  |  | 			return false; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (obj.exclude) { | 
					
						
							|  |  |  | 		if (ModuleFilenameHelpers.matchPart(str, obj.exclude)) { | 
					
						
							|  |  |  | 			return false; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-03-21 02:00:39 +08:00
										 |  |  | 	return true; | 
					
						
							|  |  |  | }; |