更新
This commit is contained in:
+165
@@ -0,0 +1,165 @@
|
||||
/*
|
||||
MIT License http://www.opensource.org/licenses/mit-license.php
|
||||
Author Tobias Koppers @sokra
|
||||
*/
|
||||
|
||||
"use strict";
|
||||
|
||||
const { cachedJoin } = require("./util/path");
|
||||
|
||||
/** @typedef {import("./Resolver")} Resolver */
|
||||
/** @typedef {import("./Resolver").ResolveRequest} ResolveRequest */
|
||||
/** @typedef {import("./Resolver").ResolveStepHook} ResolveStepHook */
|
||||
/** @typedef {import("./Resolver").ResolveContextYield} ResolveContextYield */
|
||||
/** @typedef {{ [k: string]: undefined | ResolveRequest | ResolveRequest[] }} Cache */
|
||||
|
||||
const RELATIVE_REQUEST_REGEXP = /^\.\.?(?:\/|$)/;
|
||||
|
||||
/**
|
||||
* @param {string} relativePath relative path from package root
|
||||
* @param {string} request relative request
|
||||
* @returns {string} normalized request with a preserved leading dot
|
||||
*/
|
||||
function joinRelativePreservingLeadingDot(relativePath, request) {
|
||||
const normalized = cachedJoin(relativePath, request);
|
||||
return RELATIVE_REQUEST_REGEXP.test(normalized)
|
||||
? normalized
|
||||
: `./${normalized}`;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param {ResolveRequest} request request
|
||||
* @returns {string | false | undefined} normalized path
|
||||
*/
|
||||
function getCachePath(request) {
|
||||
if (request.descriptionFileRoot && !request.module) {
|
||||
return request.descriptionFileRoot;
|
||||
}
|
||||
return request.path;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param {ResolveRequest} request request
|
||||
* @returns {string | undefined} normalized request string
|
||||
*/
|
||||
function getCacheRequest(request) {
|
||||
const requestString = request.request;
|
||||
if (
|
||||
!requestString ||
|
||||
!request.relativePath ||
|
||||
!RELATIVE_REQUEST_REGEXP.test(requestString)
|
||||
) {
|
||||
return requestString;
|
||||
}
|
||||
return joinRelativePreservingLeadingDot(request.relativePath, requestString);
|
||||
}
|
||||
|
||||
/**
|
||||
* @param {string} type type of cache
|
||||
* @param {ResolveRequest} request request
|
||||
* @param {boolean} withContext cache with context?
|
||||
* @returns {string} cache id
|
||||
*/
|
||||
function getCacheId(type, request, withContext) {
|
||||
return JSON.stringify({
|
||||
type,
|
||||
context: withContext ? request.context : "",
|
||||
path: getCachePath(request),
|
||||
query: request.query,
|
||||
fragment: request.fragment,
|
||||
request: getCacheRequest(request),
|
||||
});
|
||||
}
|
||||
|
||||
module.exports = class UnsafeCachePlugin {
|
||||
/**
|
||||
* @param {string | ResolveStepHook} source source
|
||||
* @param {(request: ResolveRequest) => boolean} filterPredicate filterPredicate
|
||||
* @param {Cache} cache cache
|
||||
* @param {boolean} withContext withContext
|
||||
* @param {string | ResolveStepHook} target target
|
||||
*/
|
||||
constructor(source, filterPredicate, cache, withContext, target) {
|
||||
this.source = source;
|
||||
this.filterPredicate = filterPredicate;
|
||||
this.withContext = withContext;
|
||||
this.cache = cache;
|
||||
this.target = target;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param {Resolver} resolver the resolver
|
||||
* @returns {void}
|
||||
*/
|
||||
apply(resolver) {
|
||||
const target = resolver.ensureHook(this.target);
|
||||
resolver
|
||||
.getHook(this.source)
|
||||
.tapAsync("UnsafeCachePlugin", (request, resolveContext, callback) => {
|
||||
if (!this.filterPredicate(request)) {
|
||||
return resolver.doResolve(
|
||||
target,
|
||||
request,
|
||||
null,
|
||||
resolveContext,
|
||||
callback,
|
||||
);
|
||||
}
|
||||
const isYield = typeof resolveContext.yield === "function";
|
||||
const cacheId = getCacheId(
|
||||
isYield ? "yield" : "default",
|
||||
request,
|
||||
this.withContext,
|
||||
);
|
||||
const cacheEntry = this.cache[cacheId];
|
||||
if (cacheEntry) {
|
||||
if (isYield) {
|
||||
const yield_ =
|
||||
/** @type {ResolveContextYield} */
|
||||
(resolveContext.yield);
|
||||
if (Array.isArray(cacheEntry)) {
|
||||
for (const result of cacheEntry) yield_(result);
|
||||
} else {
|
||||
yield_(cacheEntry);
|
||||
}
|
||||
return callback(null, null);
|
||||
}
|
||||
return callback(null, /** @type {ResolveRequest} */ (cacheEntry));
|
||||
}
|
||||
|
||||
/** @type {ResolveContextYield | undefined} */
|
||||
let yieldFn;
|
||||
/** @type {ResolveContextYield | undefined} */
|
||||
let yield_;
|
||||
/** @type {ResolveRequest[]} */
|
||||
const yieldResult = [];
|
||||
if (isYield) {
|
||||
yieldFn = resolveContext.yield;
|
||||
yield_ = (result) => {
|
||||
yieldResult.push(result);
|
||||
};
|
||||
}
|
||||
|
||||
resolver.doResolve(
|
||||
target,
|
||||
request,
|
||||
null,
|
||||
yield_ ? { ...resolveContext, yield: yield_ } : resolveContext,
|
||||
(err, result) => {
|
||||
if (err) return callback(err);
|
||||
if (isYield) {
|
||||
if (result) yieldResult.push(result);
|
||||
for (const result of yieldResult) {
|
||||
/** @type {ResolveContextYield} */
|
||||
(yieldFn)(result);
|
||||
}
|
||||
this.cache[cacheId] = yieldResult;
|
||||
return callback(null, null);
|
||||
}
|
||||
if (result) return callback(null, (this.cache[cacheId] = result));
|
||||
callback();
|
||||
},
|
||||
);
|
||||
});
|
||||
}
|
||||
};
|
||||
Reference in New Issue
Block a user