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 🙏

© 2024 – Pkg Stats / Ryan Hefner

@handy-common-utils/promise-utils

v1.6.0

Published

Promise related utilities

Downloads

3,376

Readme

@handy-common-utils/promise-utils

These Promise-related utilities boast 100% test coverage, ensuring robust reliability. The package, free of external dependencies, offers essential functions such as:

  • repeat: Executes an operation repeatedly, very useful to collect all results through pagination.
  • withRetry: Retries an operation until a specified condition is met.
  • withConcurrency: Executes multiple operations with specified level of concurrency, and abort remaining operations when an error happens.
  • inParallel: Executes multiple operations with specified level of concurrency, all operations are guaranteed to be executed regardless of any possible error.
  • delayedResolve: Creates a Promise that resolves after a specified delay.
  • delayedReject: Creates a Promise that rejects after a specified delay.
  • timeoutResolve: Applies a timeout to a Promise and resolves with a specified result if the timeout occurs.
  • timeoutReject: Applies a timeout to a Promise and rejects with a specified error/reason if the timeout occurs.
  • promiseState: Retrieves the state of a Promise.
  • synchronized: Provides mutual exclusion for concurrent operations using a lock mechanism, similar to synchronized in Java.

Version Downloads/week CI codecov

How to use

First add it as a dependency:

npm install @handy-common-utils/promise-utils

Then you can use it in the code:

import { PromiseUtils } from '@handy-common-utils/promise-utils';

// delayedResolve(...), delayedReject(...), promiseState(...)
const p1 = PromiseUtils.delayedResolve(50, 1);
const p2 = PromiseUtils.delayedReject(50, 2);
await expect(PromiseUtils.promiseState(p1)).eventually.eq(PromiseState.Pending);
await expect(PromiseUtils.promiseState(p2)).eventually.eq(PromiseState.Pending);
await PromiseUtils.delayedResolve(80);
await expect(PromiseUtils.promiseState(p1)).eventually.eq(PromiseState.Fulfilled);
await expect(PromiseUtils.promiseState(p2)).eventually.eq(PromiseState.Rejected);

// timeoutReject(...)
const p = PromiseUtils.timeoutReject(PromiseUtils.delayedReject(80, '1'), 10, '2');
await expect(p).to.be.rejectedWith('2');

// repeat(...)
async repeatFetchingItemsByPosition<T>(
  fetchItemsByPosition: (parameter: { position?: string }) => Promise<{ position?: string; items?: Array<T> }>,
) {
  return PromiseUtils.repeat(
    fetchItemsByPosition,
    response => response.position ? { position: response.position } : null,
    (collection, response) => response.items ? collection.concat(response.items) : collection,
    [] as Array<T>,
  );
}

You can either import and use the PromiseUtils class as shown above, or you can import its re-exported functions directly like below:

import { withRetry, inParallel, FIBONACCI_SEQUENCE, EXPONENTIAL_SEQUENCE } from '@handy-common-utils/promise-utils';

// withRetry(...)
const result = await withRetry(() => doSomething(), [100, 200, 300, 500, 800, 1000]);
const result2 = await withRetry(() => doSomething(), Array.from({length: 10}, (_v, i) => 1000 * Math.min(FIBONACCI_SEQUENCE[i], 10)), err => err.statusCode === 429);
const result3 = await withRetry(() => doSomething(), attempt => attempt <= 8 ? 1000 * Math.min(EXPONENTIAL_SEQUENCE[attempt - 1], 10) : undefined, err => err.statusCode === 429);
statusCode === 429);

// Capture errors in the returned array
const attributesAndPossibleErrors = await PromiseUtils.inParallel(5, topicArns, async (topicArn) => {
  const topicAttributes = (await sns.getTopicAttributes({ TopicArn: topicArn }).promise()).Attributes!;
  return topicAttributes;
});

// Abort on the first error
let results: Array<JobResult>;
try {
  results = await PromiseUtils.withConcurrency(100, jobs, async (job) => processor.process(job));
} catch (error) {
  // handle the error
}

API

@handy-common-utils/promise-utils

Enumerations

Classes

Variables

EXPONENTIAL_SEQUENCE

Const EXPONENTIAL_SEQUENCE: number[]

Array of 25 exponential numbers starting from 1 up to 33554432. It can be used to form your own backoff interval array.

Example

// 1ms, 2ms, 4ms, 8ms, 16ms, 32ms
PromiseUtils.withRetry(() => doSomething(), EXPONENTIAL_SEQUENCE.slice(0, 5), err => err.statusCode === 429);
// 1s, 2s, 4s, 8s, 10s, 10s, 10s, 10s, 10s, 10s
PromiseUtils.withRetry(() => doSomething(), Array.from({length: 10}, (_v, i) => 1000 * Math.min(EXPONENTIAL_SEQUENCE[i], 10)), err => err.statusCode === 429);
// with +-10% randomness: 1s, 2s, 4s, 8s
PromiseUtils.withRetry(() => doSomething(), FIBONACCI_SEQUENCE.slice(0, 4).map(n => 1000 * n * (1 + (Math.random() - 0.5) / 5)), err => err.statusCode === 429);

FIBONACCI_SEQUENCE

Const FIBONACCI_SEQUENCE: number[]

Array of 25 Fibonacci numbers starting from 1 up to 317811. It can be used to form your own backoff interval array.

Example

// 1ms, 2ms, 3ms, 5ms, 8ms, 13ms
PromiseUtils.withRetry(() => doSomething(), FIBONACCI_SEQUENCE.slice(0, 5), err => err.statusCode === 429);
// 1s, 2s, 3s, 4s, 8s, 10s, 10s, 10s, 10s, 10s
PromiseUtils.withRetry(() => doSomething(), Array.from({length: 10}, (_v, i) => 1000 * Math.min(FIBONACCI_SEQUENCE[i], 10)), err => err.statusCode === 429);
// with +-10% randomness: 1s, 2s, 3s, 5s, 8s, 13s
PromiseUtils.withRetry(() => doSomething(), FIBONACCI_SEQUENCE.slice(0, 5).map(n => 1000 * n * (1 + (Math.random() - 0.5) / 5)), err => err.statusCode === 429);

Functions

delayedReject

delayedReject<T, R>(ms, reason): Promise<T>

See delayedReject for full documentation.

Type parameters

| Name | Type | | :------ | :------ | | T | never | | R | any |

Parameters

| Name | Type | | :------ | :------ | | ms | number | | reason | R | PromiseLike<R> | () => R | PromiseLike<R> |

Returns

Promise<T>


delayedResolve

delayedResolve<T>(ms, result?): Promise<T>

See delayedResolve for full documentation.

Type parameters

| Name | | :------ | | T |

Parameters

| Name | Type | | :------ | :------ | | ms | number | | result? | T | PromiseLike<T> | () => T | PromiseLike<T> |

Returns

Promise<T>


inParallel

inParallel<Data, Result, TError>(parallelism, jobs, operation, options?): Promise<(Result | TError)[]>

See inParallel for full documentation.

Type parameters

| Name | Type | | :------ | :------ | | Data | Data | | Result | Result | | TError | Result |

Parameters

| Name | Type | | :------ | :------ | | parallelism | number | | jobs | Iterable<Data> | | operation | (job: Data, index: number) => Promise<Result> | | options? | Object | | options.abortOnError | boolean |

Returns

Promise<(Result | TError)[]>


promiseState

promiseState(p): Promise<PromiseState>

See promiseState for full documentation.

Parameters

| Name | Type | | :------ | :------ | | p | Promise<any> |

Returns

Promise<PromiseState>


repeat

repeat<Result, Param, Collection>(operation, nextParameter, collect, initialCollection, initialParameter?): Promise<Collection>

See repeat for full documentation.

Type parameters

| Name | | :------ | | Result | | Param | | Collection |

Parameters

| Name | Type | | :------ | :------ | | operation | (parameter: Partial<Param>) => Promise<Result> | | nextParameter | (response: Result) => null | Partial<Param> | Promise<Partial<Param>> | | collect | (collection: Collection, result: Result) => Collection | | initialCollection | Collection | | initialParameter | Partial<Param> |

Returns

Promise<Collection>


synchronised

synchronised<T>(lock, operation): Promise<T>

See synchronised for full documentation.

Type parameters

| Name | | :------ | | T |

Parameters

| Name | Type | | :------ | :------ | | lock | any | | operation | (previousState: undefined | PromiseState, previousSettledState: undefined | PromiseState, previousResult: any) => Promise<T> |

Returns

Promise<T>


synchronized

synchronized<T>(lock, operation): Promise<T>

See synchronized for full documentation.

Type parameters

| Name | | :------ | | T |

Parameters

| Name | Type | | :------ | :------ | | lock | any | | operation | (previousState: undefined | PromiseState, previousSettledState: undefined | PromiseState, previousResult: any) => Promise<T> |

Returns

Promise<T>


timeoutReject

timeoutReject<T, R>(operation, ms, rejectReason): Promise<T>

See timeoutReject for full documentation.

Type parameters

| Name | Type | | :------ | :------ | | T | never | | R | any |

Parameters

| Name | Type | | :------ | :------ | | operation | Promise<T> | () => Promise<T> | | ms | number | | rejectReason | R | PromiseLike<R> | () => R | PromiseLike<R> |

Returns

Promise<T>


timeoutResolve

timeoutResolve<T>(operation, ms, result?): Promise<T>

See timeoutResolve for full documentation.

Type parameters

| Name | | :------ | | T |

Parameters

| Name | Type | | :------ | :------ | | operation | Promise<T> | () => Promise<T> | | ms | number | | result? | T | PromiseLike<T> | () => T | PromiseLike<T> |

Returns

Promise<T>


withConcurrency

withConcurrency<Data, Result>(concurrency, jobs, operation): Promise<Result[]>

See withConcurrency for full documentation.

Type parameters

| Name | | :------ | | Data | | Result |

Parameters

| Name | Type | | :------ | :------ | | concurrency | number | | jobs | Iterable<Data> | | operation | (job: Data, index: number) => Promise<Result> |

Returns

Promise<Result[]>


withRetry

withRetry<Result, TError>(operation, backoff, shouldRetry?): Promise<Result>

See withRetry for full documentation.

Type parameters

| Name | Type | | :------ | :------ | | Result | Result | | TError | any |

Parameters

| Name | Type | | :------ | :------ | | operation | (attempt: number, previousResult: undefined | Result, previousError: undefined | TError) => Promise<Result> | | backoff | number[] | (attempt: number, previousResult: undefined | Result, previousError: undefined | TError) => undefined | number | | shouldRetry | (previousError: undefined | TError, previousResult: undefined | Result, attempt: number) => boolean |

Returns

Promise<Result>

Classes

Class: PromiseUtils

Constructors

constructor

new PromiseUtils()

Methods

delayedReject

Static delayedReject<T, R>(ms, reason): Promise<T>

Creates a Promise that rejects after a specified number of milliseconds.

Type parameters

| Name | Type | | :------ | :------ | | T | never | | R | any |

Parameters

| Name | Type | Description | | :------ | :------ | :------ | | ms | number | The number of milliseconds after which the created Promise will reject. | | reason | R | PromiseLike<R> | () => R | PromiseLike<R> | The reason for the rejection, or a function that supplies the reason. If the reason is a rejected Promise, the outcome of it will be the rejection reason of the returned Promise. |

Returns

Promise<T>

A new Promise that rejects with the specified reason after the specified delay.


delayedResolve

Static delayedResolve<T>(ms, result?): Promise<T>

Creates a Promise that resolves after a specified number of milliseconds.

Type parameters

| Name | | :------ | | T |

Parameters

| Name | Type | Description | | :------ | :------ | :------ | | ms | number | The number of milliseconds after which the created Promise will resolve. | | result? | T | PromiseLike<T> | () => T | PromiseLike<T> | The result to be resolved by the Promise, or a function that supplies the result. |

Returns

Promise<T>

A new Promise that resolves with the specified result after the specified delay.


inParallel

Static inParallel<Data, Result, TError>(parallelism, jobs, operation, options?): Promise<(Result | TError)[]>

Executes multiple jobs/operations in parallel. By default, all operations are executed regardless of any failures. In most cases, using withConcurrency might be more convenient.

By default, this function does not throw or reject an error when any job/operation fails. Errors from operations are returned alongside results in the returned array. This function only resolves when all jobs/operations are settled (either resolved or rejected).

If options.abortOnError is set to true, this function throws (or rejects with) an error immediately when any job/operation fails. In this mode, some jobs/operations may not be executed if one fails.

Type parameters

| Name | Type | Description | | :------ | :------ | :------ | | Data | Data | The type of the job data, typically an Array. | | Result | Result | The type of the return value from the operation function. | | TError | Result | The type for the error that could be thrown from the operation function, defaults to Result. |

Parameters

| Name | Type | Description | | :------ | :------ | :------ | | parallelism | number | The number of jobs/operations to run concurrently. | | jobs | Iterable<Data> | The job data to be processed. This function can safely handle an infinite or unknown number of elements. | | operation | (job: Data, index: number) => Promise<Result> | The function that processes job data asynchronously. | | options? | Object | Options to control the function's behavior. | | options.abortOnError | boolean | If true, the function aborts and throws an error on the first failed operation. |

Returns

Promise<(Result | TError)[]>

A promise that resolves to an array containing the results of the operations. Each element is either a fulfilled result or a rejected error/reason. The results or errors in the returned array are in the same order as the corresponding elements in the jobs array.

Example

// Capture errors in the returned array
const attributesAndPossibleErrors: Array<JobResult|JobError> = await PromiseUtils.inParallel(5, topicArns, async (topicArn) => {
  const topicAttributes = (await sns.getTopicAttributes({ TopicArn: topicArn }).promise()).Attributes!;
  return topicAttributes;
});

// Abort on the first error
let results: Array<JobResult>;
try {
  results = await PromiseUtils.inParallel(100, jobs, async (job) => processor.process(job), { abortOnError: true });
} catch (error) {
  // handle the error
}

promiseState

Static promiseState(p): Promise<PromiseState>

Retrieves the state of the specified Promise. Note: The returned value is a Promise that resolves immediately.

Parameters

| Name | Type | Description | | :------ | :------ | :------ | | p | Promise<any> | The Promise whose state is to be determined. |

Returns

Promise<PromiseState>

A Promise that resolves immediately with the state of the input Promise.


repeat

Static repeat<Result, Param, Collection>(operation, nextParameter, collect, initialCollection, initialParameter?): Promise<Collection>

Executes an operation repeatedly and collects all the results. This function is very useful for many scenarios, such like client-side pagination.

Type parameters

| Name | Description | | :------ | :------ | | Result | The type of the operation result. | | Param | The type of the input to the operation, typically a paging parameter. | | Collection | The type of the collection returned by this function. |

Parameters

| Name | Type | Description | | :------ | :------ | :------ | | operation | (parameter: Partial<Param>) => Promise<Result> | A function that takes a parameter as input and returns a result. Typically, the parameter has optional fields to control paging. | | nextParameter | (response: Result) => null | Partial<Param> | Promise<Partial<Param>> | A function for calculating the next parameter from the operation result. Normally, this parameter controls paging. This function should return null when no further invocation of the operation function is desired. If further invocation is desired, the return value of this function can be a Promise or a non-Promise value. | | collect | (collection: Collection, result: Result) => Collection | A function for merging the operation result into the collection. | | initialCollection | Collection | The initial collection, which will be the first argument passed to the first invocation of the collect function. | | initialParameter | Partial<Param> | The parameter for the first operation. |

Returns

Promise<Collection>

A promise that resolves to a collection of all the results returned by the operation function.

Example

const domainNameObjects = await PromiseUtils.repeat(
  pagingParam => apig.getDomainNames({limit: 500, ...pagingParam}).promise(),
  response => response.position? {position: response.position} : null,
  (collection, response) => collection.concat(response.items!),
  [] as APIGateway.DomainName[],
);

synchronised

Static synchronised<T>(lock, operation): Promise<T>

This is just another spelling of synchronized.

Type parameters

| Name | | :------ | | T |

Parameters

| Name | Type | Description | | :------ | :------ | :------ | | lock | any | The object (such as a string, a number, or this in a class) used to identify the lock. | | operation | (previousState: undefined | PromiseState, previousSettledState: undefined | PromiseState, previousResult: any) => Promise<T> | The function that performs the computation and returns a Promise. |

Returns

Promise<T>

The result of the operation function.


synchronized

Static synchronized<T>(lock, operation): Promise<T>

Provides mutual exclusion similar to synchronized in Java. Ensures no concurrent execution of any operation function associated with the same lock. The operation function has access to the state (when synchronized is called), settledState (when the operation function is called), and result (either the fulfilled result or the rejected reason) of the previous operation. If there is no previous invocation, state, settledState, and result will all be undefined.

Type parameters

| Name | | :------ | | T |

Parameters

| Name | Type | Description | | :------ | :------ | :------ | | lock | any | The object (such as a string, a number, or this in a class) used to identify the lock. | | operation | (previousState: undefined | PromiseState, previousSettledState: undefined | PromiseState, previousResult: any) => Promise<T> | The function that performs the computation and returns a Promise. |

Returns

Promise<T>

The result of the operation function.


timeoutReject

Static timeoutReject<T, R>(operation, ms, rejectReason): Promise<T>

Applies a timeout to a Promise or a function that returns a Promise. If the timeout occurs, the returned Promise rejects with the specified reason. If the timeout does not occur, the returned Promise resolves or rejects based on the outcome of the original Promise. If the rejectReason parameter is a function and the timeout does not occur, the function will not be called. Note: The rejection of the operation parameter is not handled by this function. You may want to handle it outside this function to avoid warnings like "(node:4330) PromiseRejectionHandledWarning: Promise rejection was handled asynchronously."

Type parameters

| Name | Type | | :------ | :------ | | T | never | | R | any |

Parameters

| Name | Type | Description | | :------ | :------ | :------ | | operation | Promise<T> | () => Promise<T> | The original Promise or a function that returns a Promise to which the timeout will be applied. | | ms | number | The number of milliseconds for the timeout. | | rejectReason | R | PromiseLike<R> | () => R | PromiseLike<R> | The reason to reject with if the timeout occurs, or a function that supplies the reason. |

Returns

Promise<T>

A new Promise that rejects with the specified reason if the timeout occurs.


timeoutResolve

Static timeoutResolve<T>(operation, ms, result?): Promise<T>

Applies a timeout to a Promise or a function that returns a Promise. If the timeout occurs, the returned Promise resolves to the specified result. If the timeout does not occur, the returned Promise resolves or rejects based on the outcome of the original Promise. If the result parameter is a function and the timeout does not occur, the function will not be called. Note: The rejection of the operation parameter is not handled by this function. You may want to handle it outside this function to avoid warnings like "(node:4330) PromiseRejectionHandledWarning: Promise rejection was handled asynchronously."

Type parameters

| Name | | :------ | | T |

Parameters

| Name | Type | Description | | :------ | :------ | :------ | | operation | Promise<T> | () => Promise<T> | The original Promise or a function that returns a Promise to which the timeout will be applied. | | ms | number | The number of milliseconds for the timeout. | | result? | T | PromiseLike<T> | () => T | PromiseLike<T> | The result to resolve with if the timeout occurs, or a function that supplies the result. |

Returns

Promise<T>

A new Promise that resolves to the specified result if the timeout occurs.


withConcurrency

Static withConcurrency<Data, Result>(concurrency, jobs, operation): Promise<Result[]>

Executes multiple jobs/operations with a specified level of concurrency.

Unlike inParallel(...), this function may throw or reject an error when a job/operation fails. When an error is re-thrown, remaining operations will not be executed. If you want all the operations to always be executed, use inParallel instead.

Type parameters

| Name | Description | | :------ | :------ | | Data | The type of the job data, typically an Array. | | Result | The type of the return value from the operation function. |

Parameters

| Name | Type | Description | | :------ | :------ | :------ | | concurrency | number | The number of jobs/operations to run concurrently. | | jobs | Iterable<Data> | The job data to be processed. This function can handle an infinite or unknown number of elements safely. | | operation | (job: Data, index: number) => Promise<Result> | The function that processes job data asynchronously. |

Returns

Promise<Result[]>

A promise that resolves to an array containing the results from the operation function. The results in the returned array are in the same order as the corresponding elements in the jobs array.

Example

// At any time, there would be no more than 5 concurrency API calls. Error would be re-thrown immediately when it occurs.
const attributes = await PromiseUtils.withConcurrency(5, topicArns, async (topicArn) => {
  const topicAttributes = (await sns.getTopicAttributes({ TopicArn: topicArn }).promise()).Attributes!;
  return topicAttributes;
});

withRetry

Static withRetry<Result, TError>(operation, backoff, shouldRetry?): Promise<Result>

Repeatedly performs an operation until a specified criteria is met.

Type parameters

| Name | Type | Description | | :------ | :------ | :------ | | Result | Result | Type of the operation result. | | TError | any | Type of the possible error that could be generated by the operation. |

Parameters

| Name | Type | Description | | :------ | :------ | :------ | | operation | (attempt: number, previousResult: undefined | Result, previousError: undefined | TError) => Promise<Result> | A function that outputs a Promise result. Typically, the operation does not use its arguments. | | backoff | number[] | (attempt: number, previousResult: undefined | Result, previousError: undefined | TError) => undefined | number | An array of retry backoff periods (in milliseconds) or a function for calculating them. If retry is desired, the specified backoff period is waited before the next call to the operation. If the array runs out of elements or the function returns undefined or a negative number, no further calls to the operation will be made. The attempt argument passed to the backoff function starts from 1, as it is called immediately after the first attempt and before the first retry. | | shouldRetry | (previousError: undefined | TError, previousResult: undefined | Result, attempt: number) => boolean | A predicate function for deciding whether another call to the operation should occur. If this argument is not defined, a retry will occur whenever the operation rejects with an error. The shouldRetry function is evaluated before the backoff. The attempt argument passed to the shouldRetry function starts from 1. |

Returns

Promise<Result>

A promise of the operation result, potentially with retries applied.

Example

const result = await PromiseUtils.withRetry(() => doSomething(), [100, 200, 300, 500, 800, 1000]);
const result2 = await PromiseUtils.withRetry(() => doSomething(), Array.from({length: 10}, (_v, i) => 1000 * Math.min(FIBONACCI_SEQUENCE[i], 10), err => err.statusCode === 429);
const result3 = await PromiseUtils.withRetry(() => doSomething(), attempt => attempt <= 8 ? 1000 * Math.min(FIBONACCI_SEQUENCE[attempt - 1], 10) : undefined, err => err.statusCode === 429);

Enums

Enumeration: PromiseState

The state of a Promise can only be one of: Pending, Fulfilled, and Rejected.

Enumeration Members

Fulfilled

Fulfilled = "Fulfilled"


Pending

Pending = "Pending"


Rejected

Rejected = "Rejected"