webpack/lib/Cache.js

166 lines
4.0 KiB
JavaScript
Raw Normal View History

/*
MIT License http://www.opensource.org/licenses/mit-license.php
Author Tobias Koppers @sokra
*/
"use strict";
const { AsyncParallelHook, AsyncSeriesBailHook, SyncHook } = require("tapable");
2019-08-09 20:49:47 +08:00
const {
makeWebpackError,
makeWebpackErrorCallback
} = require("./HookWebpackError");
2018-12-18 05:20:58 +08:00
/** @typedef {import("./WebpackError")} WebpackError */
/**
2024-06-11 21:09:50 +08:00
* @typedef {object} Etag
* @property {() => string} toString
*/
2018-12-18 05:20:58 +08:00
/**
* @template T
* @callback CallbackCache
2024-02-21 22:05:29 +08:00
* @param {WebpackError | null} err
* @param {T=} result
2018-12-18 05:20:58 +08:00
* @returns {void}
*/
/**
* @callback GotHandler
* @param {any} result
* @param {(err?: Error) => void} callback
2018-12-18 05:20:58 +08:00
* @returns {void}
*/
2024-02-22 22:20:17 +08:00
/**
* @param {number} times times
* @param {(err?: Error) => void} callback callback
* @returns {(err?: Error) => void} callback
2024-02-22 22:20:17 +08:00
*/
2024-07-31 11:31:11 +08:00
const needCalls = (times, callback) => err => {
if (--times === 0) {
return callback(err);
}
if (err && times > 0) {
times = 0;
return callback(err);
}
};
class Cache {
constructor() {
this.hooks = {
/** @type {AsyncSeriesBailHook<[string, Etag | null, GotHandler[]], any>} */
get: new AsyncSeriesBailHook(["identifier", "etag", "gotHandlers"]),
/** @type {AsyncParallelHook<[string, Etag | null, any]>} */
store: new AsyncParallelHook(["identifier", "etag", "data"]),
2019-08-09 20:46:42 +08:00
/** @type {AsyncParallelHook<[Iterable<string>]>} */
storeBuildDependencies: new AsyncParallelHook(["dependencies"]),
2018-12-09 19:54:17 +08:00
/** @type {SyncHook<[]>} */
beginIdle: new SyncHook([]),
2018-12-09 19:54:17 +08:00
/** @type {AsyncParallelHook<[]>} */
endIdle: new AsyncParallelHook([]),
2018-12-09 19:54:17 +08:00
/** @type {AsyncParallelHook<[]>} */
shutdown: new AsyncParallelHook([])
};
}
2018-12-18 05:20:58 +08:00
/**
* @template T
* @param {string} identifier the cache identifier
* @param {Etag | null} etag the etag
* @param {CallbackCache<T>} callback signals when the value is retrieved
2018-12-18 05:20:58 +08:00
* @returns {void}
*/
get(identifier, etag, callback) {
2024-02-22 22:20:17 +08:00
/** @type {GotHandler[]} */
const gotHandlers = [];
this.hooks.get.callAsync(identifier, etag, gotHandlers, (err, result) => {
if (err) {
2019-05-11 03:37:35 +08:00
callback(makeWebpackError(err, "Cache.hooks.get"));
return;
}
if (result === null) {
result = undefined;
}
if (gotHandlers.length > 1) {
const innerCallback = needCalls(gotHandlers.length, () =>
callback(null, result)
);
for (const gotHandler of gotHandlers) {
gotHandler(result, innerCallback);
}
} else if (gotHandlers.length === 1) {
gotHandlers[0](result, () => callback(null, result));
} else {
callback(null, result);
}
});
}
2018-12-18 05:20:58 +08:00
/**
* @template T
* @param {string} identifier the cache identifier
* @param {Etag | null} etag the etag
2018-12-18 05:20:58 +08:00
* @param {T} data the value to store
* @param {CallbackCache<void>} callback signals when the value is stored
2018-12-18 05:20:58 +08:00
* @returns {void}
*/
2018-10-09 20:30:59 +08:00
store(identifier, etag, data, callback) {
2019-08-09 20:49:47 +08:00
this.hooks.store.callAsync(
identifier,
etag,
data,
makeWebpackErrorCallback(callback, "Cache.hooks.store")
);
}
2019-08-09 20:46:42 +08:00
/**
* After this method has succeeded the cache can only be restored when build dependencies are
* @param {Iterable<string>} dependencies list of all build dependencies
* @param {CallbackCache<void>} callback signals when the dependencies are stored
2019-08-09 20:46:42 +08:00
* @returns {void}
*/
storeBuildDependencies(dependencies, callback) {
this.hooks.storeBuildDependencies.callAsync(
dependencies,
makeWebpackErrorCallback(callback, "Cache.hooks.storeBuildDependencies")
);
}
2018-12-18 05:20:58 +08:00
/**
* @returns {void}
*/
beginIdle() {
this.hooks.beginIdle.call();
}
2018-12-18 05:20:58 +08:00
/**
* @param {CallbackCache<void>} callback signals when the call finishes
2018-12-18 05:20:58 +08:00
* @returns {void}
*/
endIdle(callback) {
2019-08-09 20:49:47 +08:00
this.hooks.endIdle.callAsync(
makeWebpackErrorCallback(callback, "Cache.hooks.endIdle")
);
}
2018-12-18 05:20:58 +08:00
/**
* @param {CallbackCache<void>} callback signals when the call finishes
2018-12-18 05:20:58 +08:00
* @returns {void}
*/
shutdown(callback) {
2019-08-09 20:49:47 +08:00
this.hooks.shutdown.callAsync(
makeWebpackErrorCallback(callback, "Cache.hooks.shutdown")
);
}
}
2019-05-13 17:03:10 +08:00
Cache.STAGE_MEMORY = -10;
Cache.STAGE_DEFAULT = 0;
Cache.STAGE_DISK = 10;
Cache.STAGE_NETWORK = 20;
module.exports = Cache;