UNPKG

copy-webpack-plugin

Version:
128 lines (121 loc) 3.09 kB
"use strict"; /** @typedef {import("webpack").Compilation["inputFileSystem"] } InputFileSystem */ /** @typedef {import("fs").Stats } Stats */ /** * @param {InputFileSystem} inputFileSystem * @param {string} path * @return {Promise<undefined | Stats>} */ function stat(inputFileSystem, path) { return new Promise((resolve, reject) => { inputFileSystem.stat(path, /** * @param {null | undefined | NodeJS.ErrnoException} err * @param {undefined | Stats} stats */ // @ts-ignore (err, stats) => { if (err) { reject(err); return; } resolve(stats); }); }); } /** * @param {InputFileSystem} inputFileSystem * @param {string} path * @return {Promise<string | Buffer>} */ function readFile(inputFileSystem, path) { return new Promise((resolve, reject) => { inputFileSystem.readFile(path, /** * @param {null | undefined | NodeJS.ErrnoException} err * @param {undefined | string | Buffer} data */ (err, data) => { if (err) { reject(err); return; } resolve(/** @type {string | Buffer} */data); }); }); } const notSettled = Symbol(`not-settled`); /** * @template T * @typedef {() => Promise<T>} Task */ /** * Run tasks with limited concurrency. * @template T * @param {number} limit - Limit of tasks that run at once. * @param {Task<T>[]} tasks - List of tasks to run. * @returns {Promise<T[]>} A promise that fulfills to an array of the results */ function throttleAll(limit, tasks) { if (!Number.isInteger(limit) || limit < 1) { throw new TypeError(`Expected \`limit\` to be a finite number > 0, got \`${limit}\` (${typeof limit})`); } if (!Array.isArray(tasks) || !tasks.every(task => typeof task === `function`)) { throw new TypeError(`Expected \`tasks\` to be a list of functions returning a promise`); } return new Promise((resolve, reject) => { const result = Array(tasks.length).fill(notSettled); const entries = tasks.entries(); const next = () => { const { done, value } = entries.next(); if (done) { const isLast = !result.includes(notSettled); if (isLast) { resolve(/** @type{T[]} **/result); } return; } const [index, task] = value; /** * @param {T} x */ const onFulfilled = x => { result[index] = x; next(); }; task().then(onFulfilled, reject); }; Array(limit).fill(0).forEach(next); }); } /** * @template T * @param fn {(function(): any) | undefined} * @returns {function(): T} */ function memoize(fn) { let cache = false; /** @type {T} */ let result; return () => { if (cache) { return result; } result = /** @type {function(): any} */fn(); cache = true; // Allow to clean up memory for fn // and all dependent resources // eslint-disable-next-line no-undefined, no-param-reassign fn = undefined; return result; }; } module.exports = { stat, readFile, throttleAll, memoize };