| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369 | /*	MIT License http://www.opensource.org/licenses/mit-license.php	Author Tobias Koppers @sokra*/"use strict";const NormalModule = require("./NormalModule");const { DEFAULTS } = require("./config/defaults");const createHash = require("./util/createHash");const memoize = require("./util/memoize");/** @typedef {import("../declarations/WebpackOptions").DevtoolModuleFilenameTemplate} DevtoolModuleFilenameTemplate *//** @typedef {import("../declarations/WebpackOptions").HashFunction} HashFunction *//** @typedef {import("./ChunkGraph")} ChunkGraph *//** @typedef {import("./Module")} Module *//** @typedef {import("./RequestShortener")} RequestShortener *//** @typedef {string | RegExp | (string | RegExp)[]} Matcher *//** @typedef {{ test?: Matcher, include?: Matcher, exclude?: Matcher }} MatchObject */const ModuleFilenameHelpers = module.exports;// TODO webpack 6: consider removing theseModuleFilenameHelpers.ALL_LOADERS_RESOURCE = "[all-loaders][resource]";ModuleFilenameHelpers.REGEXP_ALL_LOADERS_RESOURCE =	/\[all-?loaders\]\[resource\]/gi;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]";// cSpell:words oluteModuleFilenameHelpers.REGEXP_ABSOLUTE_RESOURCE_PATH =	/\[abs(olute)?-?resource-?path\]/gi;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;ModuleFilenameHelpers.NAMESPACE = "[namespace]";ModuleFilenameHelpers.REGEXP_NAMESPACE = /\[namespace\]/gi;/** @typedef {() => string} ReturnStringCallback *//** * Returns a function that returns the part of the string after the token * @param {ReturnStringCallback} strFn the function to get the string * @param {string} token the token to search for * @returns {ReturnStringCallback} a function that returns the part of the string after the token */const getAfter = (strFn, token) => () => {	const str = strFn();	const idx = str.indexOf(token);	return idx < 0 ? "" : str.slice(idx);};/** * Returns a function that returns the part of the string before the token * @param {ReturnStringCallback} strFn the function to get the string * @param {string} token the token to search for * @returns {ReturnStringCallback} a function that returns the part of the string before the token */const getBefore = (strFn, token) => () => {	const str = strFn();	const idx = str.lastIndexOf(token);	return idx < 0 ? "" : str.slice(0, idx);};/** * Returns a function that returns a hash of the string * @param {ReturnStringCallback} strFn the function to get the string * @param {HashFunction=} hashFunction the hash function to use * @returns {ReturnStringCallback} a function that returns the hash of the string */const getHash =	(strFn, hashFunction = DEFAULTS.HASH_FUNCTION) =>	() => {		const hash = createHash(hashFunction);		hash.update(strFn());		const digest = /** @type {string} */ (hash.digest("hex"));		return digest.slice(0, 4);	};/** * @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. * @param {Record<string, () => T>} obj the object to convert to a lazy access object * @returns {T} the lazy access object */const lazyObject = obj => {	const newObj = /** @type {T} */ ({});	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;};const SQUARE_BRACKET_TAG_REGEXP = /\[\\*([\w-]+)\\*\]/gi;/** @typedef {((context: TODO) => string)} ModuleFilenameTemplateFunction *//** @typedef {string | ModuleFilenameTemplateFunction} ModuleFilenameTemplate *//** * @param {Module | string} module the module * @param {{ namespace?: string, moduleFilenameTemplate?: ModuleFilenameTemplate }} options options * @param {{ requestShortener: RequestShortener, chunkGraph: ChunkGraph, hashFunction?: HashFunction }} contextInfo context info * @returns {string} the filename */ModuleFilenameHelpers.createFilename = (	// eslint-disable-next-line default-param-last	module = "",	options,	{ requestShortener, chunkGraph, hashFunction = DEFAULTS.HASH_FUNCTION }) => {	const opts = {		namespace: "",		moduleFilenameTemplate: "",		...(typeof options === "object"			? options			: {					moduleFilenameTemplate: options				})	};	/** @type {ReturnStringCallback} */	let absoluteResourcePath;	let hash;	/** @type {ReturnStringCallback} */	let identifier;	/** @type {ReturnStringCallback} */	let moduleId;	/** @type {ReturnStringCallback} */	let shortIdentifier;	if (typeof module === "string") {		shortIdentifier =			/** @type {ReturnStringCallback} */			(memoize(() => requestShortener.shorten(module)));		identifier = shortIdentifier;		moduleId = () => "";		absoluteResourcePath = () =>			/** @type {string} */ (module.split("!").pop());		hash = getHash(identifier, hashFunction);	} else {		shortIdentifier = memoize(() =>			module.readableIdentifier(requestShortener)		);		identifier =			/** @type {ReturnStringCallback} */			(memoize(() => requestShortener.shorten(module.identifier())));		moduleId =			/** @type {ReturnStringCallback} */			(() => chunkGraph.getModuleId(module));		absoluteResourcePath = () =>			module instanceof NormalModule				? module.resource				: /** @type {string} */ (module.identifier().split("!").pop());		hash = getHash(identifier, hashFunction);	}	const resource =		/** @type {ReturnStringCallback} */		(memoize(() => shortIdentifier().split("!").pop()));	const loaders = getBefore(shortIdentifier, "!");	const allLoaders = getBefore(identifier, "!");	const query = getAfter(resource, "?");	const resourcePath = () => {		const q = query().length;		return q === 0 ? resource() : resource().slice(0, -q);	};	if (typeof opts.moduleFilenameTemplate === "function") {		return opts.moduleFilenameTemplate(			lazyObject({				identifier,				shortIdentifier,				resource,				resourcePath: memoize(resourcePath),				absoluteResourcePath: memoize(absoluteResourcePath),				loaders: memoize(loaders),				allLoaders: memoize(allLoaders),				query: memoize(query),				moduleId: memoize(moduleId),				hash: memoize(hash),				namespace: () => opts.namespace			})		);	}	// TODO webpack 6: consider removing alternatives without dashes	/** @type {Map<string, () => string>} */	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],		["namespace", () => opts.namespace]	]);	// TODO webpack 6: consider removing weird double placeholders	return /** @type {string} */ (opts.moduleFilenameTemplate)		.replace(ModuleFilenameHelpers.REGEXP_ALL_LOADERS_RESOURCE, "[identifier]")		.replace(			ModuleFilenameHelpers.REGEXP_LOADERS_RESOURCE,			"[short-identifier]"		)		.replace(SQUARE_BRACKET_TAG_REGEXP, (match, content) => {			if (content.length + 2 === match.length) {				const replacement = replacements.get(content.toLowerCase());				if (replacement !== undefined) {					return replacement();				}			} else if (match.startsWith("[\\") && match.endsWith("\\]")) {				return `[${match.slice(2, -2)}]`;			}			return match;		});};/** * 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"] * ``` */ModuleFilenameHelpers.replaceDuplicates = (array, fn, comparator) => {	const countMap = Object.create(null);	const posMap = Object.create(null);	for (const [idx, item] of array.entries()) {		countMap[item] = countMap[item] || [];		countMap[item].push(idx);		posMap[item] = 0;	}	if (comparator) {		for (const item of Object.keys(countMap)) {			countMap[item].sort(comparator);		}	}	return array.map((item, i) => {		if (countMap[item].length > 1) {			if (comparator && countMap[item][0] === i) return item;			return fn(item, i, posMap[item]++);		}		return item;	});};/** * 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 * ``` */const matchPart = (str, test) => {	if (!test) return true;	if (Array.isArray(test)) {		return test.some(test => matchPart(str, test));	}	if (typeof test === "string") {		return str.startsWith(test);	}	return test.test(str);};ModuleFilenameHelpers.matchPart = matchPart;/** * 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 * ``` */ModuleFilenameHelpers.matchObject = (obj, str) => {	if (obj.test && !ModuleFilenameHelpers.matchPart(str, obj.test)) {		return false;	}	if (obj.include && !ModuleFilenameHelpers.matchPart(str, obj.include)) {		return false;	}	if (obj.exclude && ModuleFilenameHelpers.matchPart(str, obj.exclude)) {		return false;	}	return true;};
 |