webpack/lib/ExternalModuleFactoryPlugin.js

342 lines
10 KiB
JavaScript
Raw Normal View History

/*
MIT License http://www.opensource.org/licenses/mit-license.php
Author Tobias Koppers @sokra
*/
2018-07-30 23:08:51 +08:00
"use strict";
2019-11-14 22:01:25 +08:00
const util = require("util");
const ExternalModule = require("./ExternalModule");
2024-03-16 00:59:30 +08:00
const ContextElementDependency = require("./dependencies/ContextElementDependency");
const CssImportDependency = require("./dependencies/CssImportDependency");
const CssUrlDependency = require("./dependencies/CssUrlDependency");
2024-03-15 22:24:33 +08:00
const HarmonyImportDependency = require("./dependencies/HarmonyImportDependency");
const ImportDependency = require("./dependencies/ImportDependency");
2025-07-03 17:06:45 +08:00
const { cachedSetProperty, resolveByProperty } = require("./util/cleverMerge");
2024-10-24 11:02:20 +08:00
/** @typedef {import("../declarations/WebpackOptions").ExternalItemFunctionData} ExternalItemFunctionData */
2025-03-27 21:58:00 +08:00
/** @typedef {import("../declarations/WebpackOptions").ExternalItemObjectKnown} ExternalItemObjectKnown */
/** @typedef {import("../declarations/WebpackOptions").ExternalItemObjectUnknown} ExternalItemObjectUnknown */
/** @typedef {import("../declarations/WebpackOptions").Externals} Externals */
/** @typedef {import("./Compilation").DepConstructor} DepConstructor */
2024-03-15 22:24:33 +08:00
/** @typedef {import("./ExternalModule").DependencyMeta} DependencyMeta */
/** @typedef {import("./Module")} Module */
2025-03-27 21:58:00 +08:00
/** @typedef {import("./ModuleFactory").IssuerLayer} IssuerLayer */
2024-03-15 22:24:33 +08:00
/** @typedef {import("./NormalModuleFactory")} NormalModuleFactory */
2021-07-08 03:56:53 +08:00
const UNSPECIFIED_EXTERNAL_TYPE_REGEXP = /^[a-z0-9-]+ /;
const EMPTY_RESOLVE_OPTIONS = {};
2019-11-15 17:10:26 +08:00
// TODO webpack 6 remove this
const callDeprecatedExternals = util.deprecate(
2024-10-24 11:02:20 +08:00
/**
2025-03-27 21:58:00 +08:00
* @param {EXPECTED_FUNCTION} externalsFunction externals function
2024-10-24 11:02:20 +08:00
* @param {string} context context
* @param {string} request request
* @param {(err: Error | null | undefined, value: ExternalValue | undefined, ty: ExternalType | undefined) => void} cb cb
*/
2019-11-15 17:10:26 +08:00
(externalsFunction, context, request, cb) => {
2024-07-31 09:37:24 +08:00
// eslint-disable-next-line no-useless-call
2019-11-15 17:10:26 +08:00
externalsFunction.call(null, context, request, cb);
},
"The externals-function should be defined like ({context, request}, cb) => { ... }",
"DEP_WEBPACK_EXTERNALS_FUNCTION_PARAMETERS"
);
2025-03-27 21:58:00 +08:00
/** @typedef {ExternalItemObjectKnown & ExternalItemObjectUnknown} ExternalItemObject */
/**
* @template {ExternalItemObject} T
* @typedef {WeakMap<T, Map<IssuerLayer, Omit<T, "byLayer">>>} ExternalWeakCache
*/
/** @type {ExternalWeakCache<ExternalItemObject>} */
2021-01-06 18:14:23 +08:00
const cache = new WeakMap();
2024-08-08 02:59:26 +08:00
/**
2025-03-27 21:58:00 +08:00
* @param {ExternalItemObject} obj obj
* @param {IssuerLayer} layer layer
* @returns {Omit<ExternalItemObject, "byLayer">} result
2024-08-08 02:59:26 +08:00
*/
2021-01-06 18:14:23 +08:00
const resolveLayer = (obj, layer) => {
2025-03-27 21:58:00 +08:00
let map = cache.get(obj);
2021-01-06 18:14:23 +08:00
if (map === undefined) {
map = new Map();
2025-03-27 21:58:00 +08:00
cache.set(obj, map);
2021-01-06 18:14:23 +08:00
} else {
const cacheEntry = map.get(layer);
if (cacheEntry !== undefined) return cacheEntry;
}
const result = resolveByProperty(obj, "byLayer", layer);
map.set(layer, result);
return result;
};
2024-10-24 11:02:20 +08:00
/** @typedef {string | string[] | boolean | Record<string, string | string[]>} ExternalValue */
/** @typedef {string | undefined} ExternalType */
2024-08-08 02:59:26 +08:00
2025-06-04 02:20:37 +08:00
const PLUGIN_NAME = "ExternalModuleFactoryPlugin";
class ExternalModuleFactoryPlugin {
/**
* @param {string | undefined} type default external type
* @param {Externals} externals externals config
*/
constructor(type, externals) {
this.type = type;
this.externals = externals;
}
/**
* @param {NormalModuleFactory} normalModuleFactory the normal module factory
* @returns {void}
*/
apply(normalModuleFactory) {
const globalType = this.type;
normalModuleFactory.hooks.factorize.tapAsync(
2025-06-04 02:20:37 +08:00
PLUGIN_NAME,
(data, callback) => {
2018-02-25 09:00:20 +08:00
const context = data.context;
const contextInfo = data.contextInfo;
2018-02-25 09:00:20 +08:00
const dependency = data.dependencies[0];
const dependencyType = data.dependencyType;
2015-07-13 06:20:09 +08:00
/** @typedef {(err?: Error | null, externalModule?: ExternalModule) => void} HandleExternalCallback */
/**
2024-08-08 02:59:26 +08:00
* @param {ExternalValue} value the external config
* @param {ExternalType | undefined} type type of external
* @param {HandleExternalCallback} callback callback
* @returns {void}
*/
2018-02-25 09:00:20 +08:00
const handleExternal = (value, type, callback) => {
if (value === false) {
// Not externals, fallback to original factory
return callback();
2018-02-25 09:00:20 +08:00
}
/** @type {string | string[] | Record<string, string|string[]>} */
2024-08-02 02:36:27 +08:00
let externalConfig = value === true ? dependency.request : value;
// When no explicit type is specified, extract it from the externalConfig
if (type === undefined) {
if (
typeof externalConfig === "string" &&
UNSPECIFIED_EXTERNAL_TYPE_REGEXP.test(externalConfig)
) {
const idx = externalConfig.indexOf(" ");
type = externalConfig.slice(0, idx);
externalConfig = externalConfig.slice(idx + 1);
} else if (
Array.isArray(externalConfig) &&
externalConfig.length > 0 &&
UNSPECIFIED_EXTERNAL_TYPE_REGEXP.test(externalConfig[0])
) {
const firstItem = externalConfig[0];
const idx = firstItem.indexOf(" ");
type = firstItem.slice(0, idx);
externalConfig = [
firstItem.slice(idx + 1),
...externalConfig.slice(1)
];
}
}
const resolvedType = /** @type {string} */ (type || globalType);
2024-03-15 22:24:33 +08:00
// TODO make it pluggable/add hooks to `ExternalModule` to allow output modules own externals?
/** @type {DependencyMeta | undefined} */
let dependencyMeta;
2024-03-15 22:24:33 +08:00
if (
dependency instanceof HarmonyImportDependency ||
2024-03-16 00:59:30 +08:00
dependency instanceof ImportDependency ||
dependency instanceof ContextElementDependency
2024-03-15 22:24:33 +08:00
) {
2024-08-03 02:34:51 +08:00
const externalType =
dependency instanceof HarmonyImportDependency
? "module"
: dependency instanceof ImportDependency
? "import"
: undefined;
2024-03-15 22:24:33 +08:00
dependencyMeta = {
2024-08-03 02:34:51 +08:00
attributes: dependency.assertions,
externalType
2024-03-15 22:24:33 +08:00
};
} else if (dependency instanceof CssImportDependency) {
dependencyMeta = {
layer: dependency.layer,
supports: dependency.supports,
media: dependency.media
};
}
if (
resolvedType === "asset" &&
dependency instanceof CssUrlDependency
) {
dependencyMeta = { sourceType: "css-url" };
}
2018-02-25 09:00:20 +08:00
callback(
null,
new ExternalModule(
externalConfig,
resolvedType,
dependency.request,
dependencyMeta
)
2018-02-25 09:00:20 +08:00
);
};
/**
* @param {Externals} externals externals config
* @param {HandleExternalCallback} callback callback
* @returns {void}
*/
2018-02-25 09:00:20 +08:00
const handleExternals = (externals, callback) => {
if (typeof externals === "string") {
if (externals === dependency.request) {
return handleExternal(dependency.request, undefined, callback);
2018-02-25 09:00:20 +08:00
}
} else if (Array.isArray(externals)) {
let i = 0;
const next = () => {
2024-03-15 22:24:33 +08:00
/** @type {boolean | undefined} */
2018-02-25 09:00:20 +08:00
let asyncFlag;
2024-03-15 22:24:33 +08:00
/**
* @param {(Error | null)=} err err
* @param {ExternalModule=} module module
* @returns {void}
*/
2018-02-25 09:00:20 +08:00
const handleExternalsAndCallback = (err, module) => {
if (err) return callback(err);
if (!module) {
if (asyncFlag) {
asyncFlag = false;
return;
}
return next();
2016-10-29 17:11:44 +08:00
}
2018-02-25 09:00:20 +08:00
callback(null, module);
};
2016-10-29 17:11:44 +08:00
2018-02-25 09:00:20 +08:00
do {
asyncFlag = true;
if (i >= externals.length) return callback();
handleExternals(externals[i++], handleExternalsAndCallback);
} while (!asyncFlag);
2018-02-25 09:00:20 +08:00
asyncFlag = false;
};
2017-11-08 18:32:05 +08:00
2018-02-25 09:00:20 +08:00
next();
return;
} else if (externals instanceof RegExp) {
if (externals.test(dependency.request)) {
return handleExternal(dependency.request, undefined, callback);
}
2018-02-25 09:00:20 +08:00
} else if (typeof externals === "function") {
2024-10-24 11:02:20 +08:00
/**
* @param {Error | null | undefined} err err
* @param {ExternalValue=} value value
* @param {ExternalType=} type type
* @returns {void}
*/
const cb = (err, value, type) => {
if (err) return callback(err);
if (value !== undefined) {
handleExternal(value, type, callback);
} else {
callback();
2018-02-25 09:00:20 +08:00
}
};
if (externals.length === 3) {
// TODO webpack 6 remove this
2019-11-15 17:10:26 +08:00
callDeprecatedExternals(
externals,
context,
dependency.request,
cb
);
} else {
const promise = externals(
{
context,
request: dependency.request,
dependencyType,
contextInfo,
getResolve: options => (context, request, callback) => {
const resolveContext = {
fileDependencies: data.fileDependencies,
missingDependencies: data.missingDependencies,
contextDependencies: data.contextDependencies
};
let resolver = normalModuleFactory.getResolver(
"normal",
dependencyType
? cachedSetProperty(
data.resolveOptions || EMPTY_RESOLVE_OPTIONS,
"dependencyType",
dependencyType
2024-01-14 09:41:34 +08:00
)
: data.resolveOptions
);
if (options) resolver = resolver.withOptions(options);
if (callback) {
resolver.resolve(
{},
context,
request,
resolveContext,
2025-03-27 21:58:00 +08:00
callback
);
} else {
return new Promise((resolve, reject) => {
resolver.resolve(
{},
context,
request,
resolveContext,
(err, result) => {
if (err) reject(err);
else resolve(result);
}
);
});
}
}
},
cb
);
if (promise && promise.then) promise.then(r => cb(null, r), cb);
}
2018-02-25 09:00:20 +08:00
return;
} else if (typeof externals === "object") {
2021-01-06 18:14:23 +08:00
const resolvedExternals = resolveLayer(
externals,
2025-03-27 21:58:00 +08:00
/** @type {IssuerLayer} */
(contextInfo.issuerLayer)
2021-01-06 18:14:23 +08:00
);
if (
Object.prototype.hasOwnProperty.call(
2021-01-06 18:14:23 +08:00
resolvedExternals,
dependency.request
)
) {
return handleExternal(
2021-01-06 18:14:23 +08:00
resolvedExternals[dependency.request],
undefined,
callback
);
}
2018-02-25 09:00:20 +08:00
}
callback();
};
2017-11-08 18:32:05 +08:00
handleExternals(this.externals, callback);
2018-02-25 09:00:20 +08:00
}
);
}
}
module.exports = ExternalModuleFactoryPlugin;