npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2025 – Pkg Stats / Ryan Hefner

@anders94/throttle

v2.0.0

Published

Parallelize function execution up to a defined limit.

Readme

throttle

Parallelize function execution in Node.js up to a defined limit.

Usually you either run functions one after the other sequentially or in parallel simultaneously. However, many times it would be more helpful to run up to some number of parallelized functions but fall back to serial beyond that. This Node.js module provides a framework for this.

Install

npm i @anders94/throttle

Quick Start

Once you enqueue() functions, they immediately start to execute.

const Throttle = require('@anders94/throttle');

const throttle = new Throttle(2);

// wrap setTimeout in a promise
const timeout = (ms) => {
    return new Promise(resolve => setTimeout(resolve, ms));
}

const wait = async (delay) => {
    console.log('start', delay, 'ms timer.');
    await timeout(delay);
    console.log('end', delay, 'ms timer.');

}

await Promise.all([
    throttle.enqueue(async () => await wait(400)),
    throttle.enqueue(async () => await wait(300)),
    throttle.enqueue(async () => await wait(200)),
    throttle.enqueue(async () => await wait(100))
]);
console.log('All functions completed!');

In the above example, const throttle = new Throttle(2) initialized the throttle to run up to 2 functions at the same time. We quickly enqueued 4 functions, each waiting less and less time, so the first two start (wait 400ms and wait 300ms) but the next two are enqueued but don't start executing. The first to finish is the second function, (wait 300ms) so wait 200ms starts. Next to finish is the first function, (wait 400ms) so wait 100ms starts next which ends soon after. Last to end is wait 200ms.

Output:

start 400 ms timer.
start 300 ms timer.
end 300 ms timer.
start 200 ms timer.
end 400 ms timer.
start 100 ms timer.
end 100 ms timer.
end 200 ms timer.

API Reference

Constructor

new Throttle(limit)

Creates a new throttle instance.

  • limit {number} - Maximum number of functions to run concurrently. Default: 2

Example:

const throttle = new Throttle(3); // Allow up to 3 concurrent functions

Methods

enqueue(fn)

Adds a function to the execution queue. If there are available execution slots (below the limit), the function starts immediately. Otherwise, it waits in the queue.

  • fn {function} - Function to execute (can be sync or async)
  • Returns: {Promise} - Promise that resolves with the function's return value or rejects with any error thrown

Example:

// Basic usage with promise
const result = await throttle.enqueue(() => 'Hello World');
console.log(result); // 'Hello World'

// With async functions
const data = await throttle.enqueue(async () => {
    const response = await fetch('https://api.example.com/data');
    return response.json();
});

// Error handling
try {
    await throttle.enqueue(() => {
        throw new Error('Something went wrong');
    });
} catch (error) {
    console.error(error.message); // 'Something went wrong'
}

dequeue()

Removes and returns the next task from the queue without executing it.

  • Returns: {object|undefined} - The next queued task object containing {fn, resolve, reject}, or undefined if queue is empty

Example:

const nextTask = throttle.dequeue();
if (nextTask) {
    console.log('Dequeued a task');
    // Note: Manual execution would require calling nextTask.fn() and handling nextTask.resolve/reject
}

isEmpty()

Checks if the queue is empty.

  • Returns: {boolean} - true if no functions are queued, false otherwise

Example:

if (!throttle.isEmpty()) {
    console.log('Still have functions waiting');
}

length()

Returns the number of functions currently in the queue (waiting to execute).

  • Returns: {number} - Number of queued functions

Example:

console.log(`${throttle.length()} functions waiting in queue`);

clearQueue()

Removes all pending functions from the queue and rejects their promises. Does not affect currently running functions.

  • Returns: {number} - Number of tasks that were cleared from the queue

Example:

// Add some functions to the queue
throttle.enqueue(() => 'task 1').catch(err => console.log('Cleared:', err.message));
throttle.enqueue(() => 'task 2').catch(err => console.log('Cleared:', err.message));

// Clear all pending tasks
const cleared = throttle.clearQueue();
console.log(`${cleared} tasks were cleared`); // "2 tasks were cleared"

finish()

Called when the queue becomes empty and all functions have completed. Currently a no-op but can be overridden.

  • Returns: undefined

Example:

throttle.finish = () => {
    console.log('All functions completed!');
};

Properties

limit

The maximum number of functions that can run concurrently.

Example:

console.log(`Current limit: ${throttle.limit}`);
throttle.limit = 5; // Change the limit dynamically

running

The number of functions currently executing.

Example:

console.log(`${throttle.running} functions currently running`);

Advanced Usage

Error Handling

Functions that throw errors will reject their returned promise but won't stop the queue. The error is contained to that specific function:

// This will reject the promise but not affect other functions
throttle.enqueue(async () => {
    throw new Error('This will not stop the queue');
}).catch(error => {
    console.error('Function failed:', error.message);
});

// This will still execute normally
throttle.enqueue(() => {
    console.log('This will still execute');
}).then(result => {
    console.log('Function completed successfully');
});

Monitoring Queue Status

You can monitor the throttle state in real-time:

const logStatus = () => {
    console.log(`Running: ${throttle.running}, Queued: ${throttle.length()}`);
};

throttle.enqueue(() => logStatus());
throttle.enqueue(() => logStatus());
throttle.enqueue(() => logStatus());

Custom Completion Handling

Override the finish() method to handle queue completion:

throttle.finish = () => {
    console.log('All tasks completed successfully!');
    process.exit(0);
};

Queue Cleanup

Use clearQueue() for cleanup scenarios like application shutdown:

// Graceful shutdown
process.on('SIGTERM', () => {
    console.log('Shutting down...');
    const cleared = throttle.clearQueue();
    console.log(`Cancelled ${cleared} pending tasks`);
    process.exit(0);
});

License

MIT