123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400 |
- /*
- MIT License http://www.opensource.org/licenses/mit-license.php
- Author Tobias Koppers @sokra
- */
- "use strict";
- const NormalModule = require("./NormalModule");
- const createHash = require("./util/createHash");
- const memoize = require("./util/memoize");
- /** @typedef {import("./ChunkGraph")} ChunkGraph */
- /** @typedef {import("./Module")} Module */
- /** @typedef {import("./RequestShortener")} RequestShortener */
- /** @typedef {typeof import("./util/Hash")} Hash */
- /** @typedef {string | RegExp | (string | RegExp)[]} Matcher */
- /** @typedef {{test?: Matcher, include?: Matcher, exclude?: Matcher }} MatchObject */
- const ModuleFilenameHelpers = exports;
- // TODO webpack 6: consider removing these
- ModuleFilenameHelpers.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 olute
- ModuleFilenameHelpers.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) => {
- return () => {
- 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) => {
- return () => {
- 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 {string | Hash=} hashFunction the hash function to use
- * @returns {ReturnStringCallback} a function that returns the hash of the string
- */
- const getHash = (strFn, hashFunction = "md4") => {
- return () => {
- const hash = createHash(hashFunction);
- hash.update(strFn());
- const digest = /** @type {string} */ (hash.digest("hex"));
- return digest.slice(0, 4);
- };
- };
- /**
- * 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
- * ```
- */
- const asRegExp = test => {
- if (typeof test === "string") {
- // 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
- test = new RegExp("^" + test.replace(/[-[\]{}()*+?.,\\^$|#\s]/g, "\\$&"));
- }
- return test;
- };
- /**
- * @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 {object} the lazy access object
- */
- 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;
- };
- const SQUARE_BRACKET_TAG_REGEXP = /\[\\*([\w-]+)\\*\]/gi;
- /**
- *
- * @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
- * @param {string | Hash=} contextInfo.hashFunction the hash function to use
- * @returns {string} the filename
- */
- ModuleFilenameHelpers.createFilename = (
- module = "",
- options,
- { requestShortener, chunkGraph, hashFunction = "md4" }
- ) => {
- const opts = {
- namespace: "",
- moduleFilenameTemplate: "",
- ...(typeof options === "object"
- ? options
- : {
- moduleFilenameTemplate: options
- })
- };
- 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 = () => 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
- : 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: identifier,
- shortIdentifier: shortIdentifier,
- resource: 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, function(): 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);
- array.forEach((item, idx) => {
- countMap[item] = countMap[item] || [];
- countMap[item].push(idx);
- posMap[item] = 0;
- });
- if (comparator) {
- Object.keys(countMap).forEach(item => {
- 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]++);
- } else {
- 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
- * ```
- */
- ModuleFilenameHelpers.matchPart = (str, test) => {
- if (!test) return true;
- if (Array.isArray(test)) {
- return test.map(asRegExp).some(regExp => regExp.test(str));
- } else {
- return asRegExp(test).test(str);
- }
- };
- /**
- * 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) {
- 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;
- }
- }
- return true;
- };
|