UNPKG

ts-prime

Version:

A utility library for JavaScript and Typescript.

55 lines (48 loc) 2.03 kB
import { ArgsType } from "./cache" import { delay } from "./delay" export interface RateLimiterOptions<F extends (...args: unknown[]) => Promise<unknown>> { rateLimitId: (...args: ArgsType<F>) => string concurrentRequests?: ((rateLimitId: string) => number) | number maxTotalRequests?: number } /** * Controls how many concurrent execution can be invoked. At any given time * @param errorHandler logic when to retry * @param request request function * @signature * P.concurrent(fn, options) * @example * const requestToEndpoint = async (endpoint: "A" | "B" | "C" | "D", data: any) => { ... } * const rate = P.concurrent(requestToEndpoint, ({ rateLimitId: (endpoint) => endpoint, concurrentRequests: 2 }) * // Only two request are fired to endpoint A * const endpoints = await Promise.all([{ endpoint: A, data: any }, ...].map(async (obj)=>{ * return rate(obj.endpoint, obj.data) * })) * @category Utility, Promise */ export function concurrent<F extends (...args: any[]) => Promise<unknown>>(request: F, options: RateLimiterOptions<F>): F { const requestCount: Record<string, number> = {} const stats = { totalRequests: 0 } const requestMiddleware = (async (...args) => { while (stats.totalRequests >= (options.maxTotalRequests || 50)) { await delay(0) } const namespace = options.rateLimitId(...args as ArgsType<F>) if (requestCount[namespace] == null) { requestCount[namespace] = 0 } const maxRequests = typeof options.concurrentRequests === 'function' ? options.concurrentRequests(namespace) : options.concurrentRequests ?? 3 while (requestCount[namespace] >= maxRequests) { await delay(0) } requestCount[namespace] += 1 stats.totalRequests += 1 return request(...args).finally(() => { requestCount[namespace] -= 1 stats.totalRequests -= 1 }) }) as F return requestMiddleware }