webpack/lib/ModuleFilenameHelpers.js

385 lines
14 KiB
JavaScript
Raw Normal View History

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");
const { DEFAULTS } = require("./config/defaults");
const createHash = require("./util/createHash");
const memoize = require("./util/memoize");
2025-04-26 01:43:01 +08:00
/** @typedef {import("../declarations/WebpackOptions").HashFunction} HashFunction */
/** @typedef {import("./ChunkGraph")} ChunkGraph */
/** @typedef {import("./Module")} Module */
/** @typedef {import("./RequestShortener")} RequestShortener */
2023-05-17 19:25:07 +08:00
/** @typedef {string | RegExp | (string | RegExp)[]} Matcher */
/** @typedef {{ test?: Matcher, include?: Matcher, exclude?: Matcher }} MatchObject */
2024-07-31 04:54:55 +08:00
const ModuleFilenameHelpers = module.exports;
2014-07-18 19:31:50 +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;
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 */
/**
* 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
* @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
*/
2024-07-31 11:31:11 +08:00
const getAfter = (strFn, token) => () => {
const str = strFn();
const idx = str.indexOf(token);
return idx < 0 ? "" : str.slice(idx);
2017-11-08 18:32:05 +08:00
};
2014-07-18 19:31:50 +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
* @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
*/
2024-07-31 11:31:11 +08:00
const getBefore = (strFn, token) => () => {
const str = strFn();
const idx = str.lastIndexOf(token);
return idx < 0 ? "" : str.slice(0, idx);
2017-11-08 18:32:05 +08:00
};
2014-07-18 19:31:50 +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
2025-04-26 01:43:01 +08:00
* @param {HashFunction=} hashFunction the hash function to use
2024-03-21 20:09:45 +08:00
* @returns {ReturnStringCallback} a function that returns the hash of the string
*/
2024-07-31 11:31:11 +08:00
const getHash =
(strFn, hashFunction = DEFAULTS.HASH_FUNCTION) =>
2024-07-31 11:31:11 +08:00
() => {
const hash = createHash(hashFunction);
2020-12-04 18:12:45 +08:00
hash.update(strFn());
const digest = /** @type {string} */ (hash.digest("hex"));
return digest.slice(0, 4);
2020-12-04 18:12:45 +08:00
};
2014-07-18 19:31:50 +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
2025-08-21 00:39:55 +08:00
* @returns {Record<string, T>} the lazy access object
*/
const lazyObject = (obj) => {
2025-08-21 00:39:55 +08:00
const newObj = /** @type {Record<string, T>} */ ({});
2020-12-04 18:12:45 +08:00
for (const key of Object.keys(obj)) {
const fn = obj[key];
Object.defineProperty(newObj, key, {
get: () => fn(),
set: (v) => {
2020-12-04 18:12:45 +08:00
Object.defineProperty(newObj, key, {
value: v,
enumerable: true,
writable: true
});
},
enumerable: true,
configurable: true
});
}
return newObj;
};
const SQUARE_BRACKET_TAG_REGEXP = /\[\\*([\w-]+)\\*\]/gi;
2025-08-21 00:39:55 +08:00
/**
* @typedef {object} ModuleFilenameTemplateContext
* @property {string} identifier the identifier of the module
* @property {string} shortIdentifier the shortened identifier of the module
* @property {string} resource the resource of the module request
* @property {string} resourcePath the resource path of the module request
* @property {string} absoluteResourcePath the absolute resource path of the module request
* @property {string} loaders the loaders of the module request
* @property {string} allLoaders the all loaders of the module request
* @property {string} query the query of the module identifier
* @property {string} moduleId the module id of the module
* @property {string} hash the hash of the module identifier
* @property {string} namespace the module namespace
*/
/** @typedef {((context: ModuleFilenameTemplateContext) => string)} ModuleFilenameTemplateFunction */
2025-04-26 01:43:01 +08:00
/** @typedef {string | ModuleFilenameTemplateFunction} ModuleFilenameTemplate */
/**
* @param {Module | string} module the module
2025-04-26 01:43:01 +08:00
* @param {{ namespace?: string, moduleFilenameTemplate?: ModuleFilenameTemplate }} options options
* @param {{ requestShortener: RequestShortener, chunkGraph: ChunkGraph, hashFunction?: HashFunction }} contextInfo context info
* @returns {string} the filename
*/
ModuleFilenameHelpers.createFilename = (
2024-07-31 09:37:24 +08:00
// eslint-disable-next-line default-param-last
module = "",
options,
{ requestShortener, chunkGraph, hashFunction = DEFAULTS.HASH_FUNCTION }
) => {
const opts = {
namespace: "",
moduleFilenameTemplate: "",
...(typeof options === "object"
2018-02-25 09:00:20 +08:00
? options
: {
moduleFilenameTemplate: options
2024-07-31 05:43:19 +08:00
})
};
2025-03-14 00:34:04 +08:00
/** @type {ReturnStringCallback} */
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 = () => "";
2025-03-14 00:34:04 +08:00
absoluteResourcePath = () =>
/** @type {string} */ (module.split("!").pop());
hash = getHash(identifier, hashFunction);
2014-07-18 19:31:50 +08:00
} else {
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));
absoluteResourcePath = () =>
2021-11-26 16:41:57 +08:00
module instanceof NormalModule
? module.resource
2025-03-14 00:34:04 +08:00
: /** @type {string} */ (module.identifier().split("!").pop());
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()));
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(
2025-08-21 00:39:55 +08:00
/** @type {ModuleFilenameTemplateContext} */
(
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
})
)
2020-12-04 18:12:45 +08:00
);
}
// 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],
2020-12-04 18:12:45 +08:00
["namespace", () => opts.namespace]
]);
// TODO webpack 6: consider removing weird double placeholders
2024-03-21 20:09:45 +08:00
return /** @type {string} */ (opts.moduleFilenameTemplate)
.replace(ModuleFilenameHelpers.REGEXP_ALL_LOADERS_RESOURCE, "[identifier]")
2018-02-25 09:00:20 +08:00
.replace(
ModuleFilenameHelpers.REGEXP_LOADERS_RESOURCE,
"[short-identifier]"
2018-02-25 09:00:20 +08:00
)
.replace(SQUARE_BRACKET_TAG_REGEXP, (match, content) => {
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();
}
} else if (match.startsWith("[\\") && match.endsWith("\\]")) {
return `[${match.slice(2, -2)}]`;
}
return match;
});
2014-07-18 19:31:50 +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
2025-04-16 22:04:11 +08:00
* @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);
2024-08-02 02:36:27 +08:00
for (const [idx, item] of array.entries()) {
2018-02-25 09:00:20 +08:00
countMap[item] = countMap[item] || [];
countMap[item].push(idx);
2014-07-18 19:31:50 +08:00
posMap[item] = 0;
2024-08-02 02:36:27 +08:00
}
2018-02-25 09:00:20 +08:00
if (comparator) {
2024-08-02 02:36:27 +08:00
for (const item of Object.keys(countMap)) {
countMap[item].sort(comparator);
2024-08-02 02:36:27 +08:00
}
}
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]++);
}
2024-07-31 04:21:27 +08:00
return item;
2014-07-18 19:31:50 +08:00
});
};
2015-03-21 02:00:39 +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
* ```
*/
const matchPart = (str, test) => {
2018-02-25 09:00:20 +08:00
if (!test) return true;
if (Array.isArray(test)) {
return test.some((test) => matchPart(str, test));
2015-03-21 02:00:39 +08:00
}
if (typeof test === "string") {
return str.startsWith(test);
}
return test.test(str);
2015-03-21 02:00:39 +08:00
};
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
* ```
*/
2017-11-08 18:32:05 +08:00
ModuleFilenameHelpers.matchObject = (obj, str) => {
2024-08-02 02:36:27 +08:00
if (obj.test && !ModuleFilenameHelpers.matchPart(str, obj.test)) {
return false;
}
2024-08-02 02:36:27 +08:00
if (obj.include && !ModuleFilenameHelpers.matchPart(str, obj.include)) {
return false;
}
2024-08-02 02:36:27 +08:00
if (obj.exclude && ModuleFilenameHelpers.matchPart(str, obj.exclude)) {
return false;
}
2015-03-21 02:00:39 +08:00
return true;
};