While the Queuer provides synchronous queueing with timing controls, the AsyncQueuer is designed specifically for handling concurrent asynchronous operations. It implements what is traditionally known as a "task pool" or "worker pool" pattern, allowing multiple operations to be processed simultaneously while maintaining control over concurrency and timing. The implementation is mostly copied from Swimmer, Tanner's original task pooling utility that has been serving the JavaScript community since 2017.
The AsyncQueuer can process multiple operations simultaneously through its task pooling mechanism. This capability is crucial for:
AsyncQueuer is designed to work seamlessly with Promises and async/await:
The concurrency limit can be adjusted at runtime using the throttle method. This allows the system to:
import { AsyncQueuer } from '@tanstack/pacer'
const asyncQueuer = new AsyncQueuer<string>({
concurrency: 2, // Process 2 items at once
wait: 1000, // Wait 1 second between starting new items
})
// Add error and success handlers
asyncQueuer.onError((error, task) => {
console.error('Task failed:', error)
})
asyncQueuer.onSuccess((result, task) => {
console.log('Task completed:', result)
})
// Start processing
asyncQueuer.start()
// Add async tasks
asyncQueuer.addItem(async () => {
const result = await fetchData(1)
return result
})
asyncQueuer.addItem(async () => {
const result = await fetchData(2)
return result
})
import { AsyncQueuer } from '@tanstack/pacer'
const asyncQueuer = new AsyncQueuer<string>({
concurrency: 2, // Process 2 items at once
wait: 1000, // Wait 1 second between starting new items
})
// Add error and success handlers
asyncQueuer.onError((error, task) => {
console.error('Task failed:', error)
})
asyncQueuer.onSuccess((result, task) => {
console.log('Task completed:', result)
})
// Start processing
asyncQueuer.start()
// Add async tasks
asyncQueuer.addItem(async () => {
const result = await fetchData(1)
return result
})
asyncQueuer.addItem(async () => {
const result = await fetchData(2)
return result
})
AsyncQueuer's error handling system is comprehensive and flexible, providing multiple ways to handle failures:
Each operation can handle its own errors through the Promise chain:
asyncQueuer
.addItem(async () => await riskyOperation())
.catch(error => handleError(error))
asyncQueuer
.addItem(async () => await riskyOperation())
.catch(error => handleError(error))
Global handlers can catch errors from any operation:
const asyncQueuer = new AsyncQueuer<string>()
// Handle task errors
asyncQueuer.onError((error, task) => {
console.error('Task failed:', error)
// Optionally retry the task or notify monitoring systems
})
// Handle successful completions
asyncQueuer.onSuccess((result, task) => {
console.log('Task completed:', result)
// Update UI or trigger dependent operations
})
// Handle all task completions (success or error)
asyncQueuer.onSettled((result, error) => {
if (error) {
console.log('Task failed:', error)
} else {
console.log('Task succeeded:', result)
}
// Update progress indicators or clean up resources
})
const asyncQueuer = new AsyncQueuer<string>()
// Handle task errors
asyncQueuer.onError((error, task) => {
console.error('Task failed:', error)
// Optionally retry the task or notify monitoring systems
})
// Handle successful completions
asyncQueuer.onSuccess((result, task) => {
console.log('Task completed:', result)
// Update UI or trigger dependent operations
})
// Handle all task completions (success or error)
asyncQueuer.onSettled((result, error) => {
if (error) {
console.log('Task failed:', error)
} else {
console.log('Task succeeded:', result)
}
// Update progress indicators or clean up resources
})
The error handling system enables sophisticated recovery strategies:
AsyncQueuer is particularly well-suited for:
Your weekly dose of JavaScript news. Delivered every Monday to over 100,000 devs, for free.