webpack/lib/ResolverFactory.js

186 lines
5.8 KiB
JavaScript
Raw Normal View History

/*
2018-07-30 23:08:51 +08:00
MIT License http://www.opensource.org/licenses/mit-license.php
Author Tobias Koppers @sokra
*/
"use strict";
2018-11-26 04:29:48 +08:00
const Factory = require("enhanced-resolve").ResolverFactory;
2018-07-30 23:08:51 +08:00
const { HookMap, SyncHook, SyncWaterfallHook } = require("tapable");
const {
cachedCleverMerge,
2020-12-30 17:42:11 +08:00
removeOperations,
resolveByProperty
} = require("./util/cleverMerge");
2020-06-17 19:22:37 +08:00
/** @typedef {import("enhanced-resolve").ResolveOptions} ResolveOptions */
/** @typedef {import("enhanced-resolve").Resolver} Resolver */
2020-06-17 19:22:37 +08:00
/** @typedef {import("../declarations/WebpackOptions").ResolveOptions} WebpackResolveOptions */
2020-07-13 23:15:20 +08:00
/** @typedef {import("../declarations/WebpackOptions").ResolvePluginInstance} ResolvePluginInstance */
2018-10-22 15:02:39 +08:00
/** @typedef {WebpackResolveOptions & { dependencyType?: string, resolveToContext?: boolean }} ResolveOptionsWithDependencyType */
/**
2024-06-11 21:09:50 +08:00
* @typedef {object} WithOptions
* @property {(options: Partial<ResolveOptionsWithDependencyType>) => ResolverWithOptions} withOptions create a resolver with additional/different options
*/
/** @typedef {Resolver & WithOptions} ResolverWithOptions */
// need to be hoisted on module level for caching identity
/** @type {ResolveOptionsWithDependencyType} */
2020-03-10 09:59:46 +08:00
const EMPTY_RESOLVE_OPTIONS = {};
/**
* @param {ResolveOptionsWithDependencyType} resolveOptionsWithDepType enhanced options
2025-09-10 03:11:22 +08:00
* @param {ResolveOptionsWithDependencyType} issuerResolveOptions enhanced options
* @returns {ResolveOptions} merged options
*/
2025-09-10 03:11:22 +08:00
const convertToResolveOptions = (
resolveOptionsWithDepType,
issuerResolveOptions
) => {
2020-12-30 17:42:11 +08:00
const { dependencyType, plugins, ...remaining } = resolveOptionsWithDepType;
// check type compat
2025-04-07 21:09:05 +08:00
/** @type {Partial<ResolveOptionsWithDependencyType>} */
2020-07-13 23:15:20 +08:00
const partialOptions = {
...remaining,
plugins:
plugins &&
2021-05-11 15:31:46 +08:00
/** @type {ResolvePluginInstance[]} */ (
plugins.filter((item) => item !== "...")
2021-05-11 15:31:46 +08:00
)
2020-07-13 23:15:20 +08:00
};
if (!partialOptions.fileSystem) {
throw new Error(
"fileSystem is missing in resolveOptions, but it's required for enhanced-resolve"
);
}
// These weird types validate that we checked all non-optional properties
2021-05-11 15:31:46 +08:00
const options =
2025-04-07 21:09:05 +08:00
/** @type {Partial<ResolveOptionsWithDependencyType> & Pick<ResolveOptionsWithDependencyType, "fileSystem">} */ (
2021-05-11 15:31:46 +08:00
partialOptions
);
2025-09-10 03:11:22 +08:00
const { issuerDependency } = issuerResolveOptions;
const resolvedByDependency = resolveByProperty(
options,
"byDependency",
dependencyType
);
2025-04-07 21:09:05 +08:00
return /** @type {ResolveOptions} */ (
2025-09-10 03:11:22 +08:00
// byIssuerDependencyType
2025-04-07 21:09:05 +08:00
removeOperations(
2025-09-10 03:11:22 +08:00
issuerDependency
? resolveByProperty(
resolvedByDependency,
"byIssuerDependency",
issuerDependency
)
: resolvedByDependency,
2025-04-07 21:09:05 +08:00
// Keep the `unsafeCache` because it can be a `Proxy`
["unsafeCache"]
)
2020-12-30 17:42:11 +08:00
);
};
2019-07-17 21:38:07 +08:00
2018-11-24 04:50:26 +08:00
/**
2024-06-11 21:09:50 +08:00
* @typedef {object} ResolverCache
* @property {WeakMap<ResolveOptionsWithDependencyType, ResolverWithOptions>} direct
* @property {Map<string, ResolverWithOptions>} stringified
2018-11-24 04:50:26 +08:00
*/
2018-06-26 14:27:44 +08:00
module.exports = class ResolverFactory {
constructor() {
2018-07-30 20:25:40 +08:00
this.hooks = Object.freeze({
/** @type {HookMap<SyncWaterfallHook<[ResolveOptionsWithDependencyType]>>} */
2018-02-25 09:00:20 +08:00
resolveOptions: new HookMap(
() => new SyncWaterfallHook(["resolveOptions"])
2018-02-25 09:00:20 +08:00
),
/** @type {HookMap<SyncHook<[Resolver, ResolveOptions, ResolveOptionsWithDependencyType]>>} */
resolver: new HookMap(
() => new SyncHook(["resolver", "resolveOptions", "userResolveOptions"])
)
2018-07-30 20:25:40 +08:00
});
2018-11-24 04:50:26 +08:00
/** @type {Map<string, ResolverCache>} */
2018-10-22 15:02:39 +08:00
this.cache = new Map();
}
2018-10-22 15:02:39 +08:00
/**
* @param {string} type type of resolver
* @param {ResolveOptionsWithDependencyType=} resolveOptions options
2025-09-10 03:11:22 +08:00
* @param {ResolveOptionsWithDependencyType=} issuerResolveOptions options
* @returns {ResolverWithOptions} the resolver
2018-10-22 15:02:39 +08:00
*/
2025-09-10 03:11:22 +08:00
get(
type,
resolveOptions = EMPTY_RESOLVE_OPTIONS,
issuerResolveOptions = EMPTY_RESOLVE_OPTIONS
) {
// let typedCaches = this.cache.get(type);
// if (!typedCaches) {
// typedCaches = {
// direct: new WeakMap(),
// stringified: new Map()
// };
// this.cache.set(type, typedCaches);
// }
// const cachedResolver = typedCaches.direct.get(resolveOptions);
// if (cachedResolver) {
// return cachedResolver;
// }
// const ident = JSON.stringify(resolveOptions);
// const resolver = typedCaches.stringified.get(ident);
// if (resolver) {
// typedCaches.direct.set(resolveOptions, resolver);
// return resolver;
// }
const newResolver = this._create(
type,
resolveOptions,
issuerResolveOptions
);
// typedCaches.direct.set(resolveOptions, newResolver);
// typedCaches.stringified.set(ident, newResolver);
return newResolver;
}
2018-10-22 15:02:39 +08:00
/**
* @param {string} type type of resolver
* @param {ResolveOptionsWithDependencyType} resolveOptionsWithDepType options
2025-09-10 03:11:22 +08:00
* @param {ResolveOptionsWithIssuerDependencyType} issuerResolveOptions options
* @returns {ResolverWithOptions} the resolver
2018-10-22 15:02:39 +08:00
*/
2025-09-10 03:11:22 +08:00
_create(type, resolveOptionsWithDepType, issuerResolveOptions) {
/** @type {ResolveOptionsWithDependencyType} */
const originalResolveOptions = { ...resolveOptionsWithDepType };
2020-06-17 19:22:37 +08:00
const resolveOptions = convertToResolveOptions(
2025-09-10 03:11:22 +08:00
this.hooks.resolveOptions.for(type).call(resolveOptionsWithDepType),
issuerResolveOptions
);
2021-05-11 15:31:46 +08:00
const resolver = /** @type {ResolverWithOptions} */ (
Factory.createResolver(resolveOptions)
);
2018-02-25 09:00:20 +08:00
if (!resolver) {
throw new Error("No resolver created");
}
/** @type {WeakMap<Partial<ResolveOptionsWithDependencyType>, ResolverWithOptions>} */
const childCache = new WeakMap();
resolver.withOptions = (options) => {
const cacheEntry = childCache.get(options);
if (cacheEntry !== undefined) return cacheEntry;
const mergedOptions = cachedCleverMerge(originalResolveOptions, options);
const resolver = this.get(type, mergedOptions);
childCache.set(options, resolver);
return resolver;
};
this.hooks.resolver
.for(type)
.call(resolver, resolveOptions, originalResolveOptions);
return resolver;
}
};