import Queue from 'yocto-queue'; export default function pLimit(concurrency) { let rejectOnClear = false; if (typeof concurrency === 'object') { ({concurrency, rejectOnClear = false} = concurrency); } validateConcurrency(concurrency); if (typeof rejectOnClear !== 'boolean') { throw new TypeError('Expected `rejectOnClear` to be a boolean'); } const queue = new Queue(); let activeCount = 0; const resumeNext = () => { // Process the next queued function if we're under the concurrency limit if (activeCount < concurrency && queue.size > 0) { activeCount++; queue.dequeue().run(); } }; const next = () => { activeCount--; resumeNext(); }; const run = async (function_, resolve, arguments_) => { // Execute the function and capture the result promise const result = (async () => function_(...arguments_))(); // Resolve immediately with the promise (don't wait for completion) resolve(result); // Wait for the function to complete (success or failure) // We catch errors here to prevent unhandled rejections, // but the original promise rejection is preserved for the caller try { await result; } catch {} // Decrement active count and process next queued function next(); }; const enqueue = (function_, resolve, reject, arguments_) => { const queueItem = {reject}; // Queue the internal resolve function instead of the run function // to preserve the asynchronous execution context. new Promise(internalResolve => { // eslint-disable-line promise/param-names queueItem.run = internalResolve; queue.enqueue(queueItem); }).then(run.bind(undefined, function_, resolve, arguments_)); // eslint-disable-line promise/prefer-await-to-then // Start processing immediately if we haven't reached the concurrency limit if (activeCount < concurrency) { resumeNext(); } }; const generator = (function_, ...arguments_) => new Promise((resolve, reject) => { enqueue(function_, resolve, reject, arguments_); }); Object.defineProperties(generator, { activeCount: { get: () => activeCount, }, pendingCount: { get: () => queue.size, }, clearQueue: { value() { if (!rejectOnClear) { queue.clear(); return; } const abortError = AbortSignal.abort().reason; while (queue.size > 0) { queue.dequeue().reject(abortError); } }, }, concurrency: { get: () => concurrency, set(newConcurrency) { validateConcurrency(newConcurrency); concurrency = newConcurrency; queueMicrotask(() => { // eslint-disable-next-line no-unmodified-loop-condition while (activeCount < concurrency && queue.size > 0) { resumeNext(); } }); }, }, map: { async value(iterable, function_) { const promises = Array.from(iterable, (value, index) => this(function_, value, index)); return Promise.all(promises); }, }, }); return generator; } export function limitFunction(function_, options) { const limit = pLimit(options); return (...arguments_) => limit(() => function_(...arguments_)); } function validateConcurrency(concurrency) { if (!((Number.isInteger(concurrency) || concurrency === Number.POSITIVE_INFINITY) && concurrency > 0)) { throw new TypeError('Expected `concurrency` to be a number from 1 and up'); } }