/** Options for `throttle` */ export interface Options { /** If it's set to `true` and queue isn't empty, the function is executed again at the end. * * default: `false` */ trailing: boolean; } /** Result of `throttle` */ export interface Result { /** whether the provided function is executed */ executed: boolean; /** the result of the provided function * * If the function is not executed, it is set to `undefined` */ result?: U; } type Resolve = (_value: T | PromiseLike) => void; type Reject = (reason?: unknown) => void; type Queue = { parameters: T; resolve: Resolve>; reject: Reject; }; /** Make the async function execute only at a time * * @param callback the async function making execute only at a time * @param options options */ export function throttle( callback: (..._args: T) => Promise, options?: Options, ): (...parameters: T) => Promise> { const { trailing = false } = options ?? {}; let queue: Queue | undefined; let running = false; const cancel = () => queue?.resolve?.({ executed: false }); const runNext = async () => { if (running || !queue) { return; } running = true; const { parameters, resolve, reject } = queue; queue = undefined; try { const result = await callback(...parameters); running = false; resolve({ result, executed: true }); } catch (e) { running = false; reject(e); } finally { if (trailing) { await runNext(); } else { cancel(); await Promise.resolve(); } } }; return (...parameters: T) => new Promise>((resolve, reject) => { queue?.resolve?.({ executed: false }); queue = { parameters, resolve, reject }; runNext(); }); }