UNPKG

@trpc/server

Version:

The tRPC server library

121 lines 7.29 kB
import type { PromiseExecutor, PromiseWithResolvers, ProxyPromise, SubscribedPromise } from "./types"; /** * Every `Promise<T>` can be shadowed by a single `ProxyPromise<T>`. It is * created once, cached and reused throughout the lifetime of the Promise. Get a * Promise's ProxyPromise using `Unpromise.proxy(promise)`. * * The `ProxyPromise<T>` attaches handlers to the original `Promise<T>` * `.then()` and `.catch()` just once. Promises derived from it use a * subscription- (and unsubscription-) based mechanism that monitors these * handlers. * * Every time you call `.subscribe()`, `.then()` `.catch()` or `.finally()` on a * `ProxyPromise<T>` it returns a `SubscribedPromise<T>` having an additional * `unsubscribe()` method. Calling `unsubscribe()` detaches reference chains * from the original, potentially long-lived Promise, eliminating memory leaks. * * This approach can eliminate the memory leaks that otherwise come about from * repeated `race()` or `any()` calls invoking `.then()` and `.catch()` multiple * times on the same long-lived native Promise (subscriptions which can never be * cleaned up). * * `Unpromise.race(promises)` is a reference implementation of `Promise.race` * avoiding memory leaks when using long-lived unsettled Promises. * * `Unpromise.any(promises)` is a reference implementation of `Promise.any` * avoiding memory leaks when using long-lived unsettled Promises. * * `Unpromise.resolve(promise)` returns an ephemeral `SubscribedPromise<T>` for * any given `Promise<T>` facilitating arbitrary async/await patterns. Behind * the scenes, `resolve` is implemented simply as * `Unpromise.proxy(promise).subscribe()`. Don't forget to call `.unsubscribe()` * to tidy up! * */ export declare class Unpromise<T> implements ProxyPromise<T> { /** INSTANCE IMPLEMENTATION */ /** The promise shadowed by this Unpromise<T> */ protected readonly promise: Promise<T> | PromiseLike<T>; /** Promises expecting eventual settlement (unless unsubscribed first). This list is deleted * after the original promise settles - no further notifications will be issued. */ protected subscribers: ReadonlyArray<PromiseWithResolvers<T>> | null; /** The Promise's settlement (recorded when it fulfils or rejects). This is consulted when * calling .subscribe() .then() .catch() .finally() to see if an immediately-resolving Promise * can be returned, and therefore subscription can be bypassed. */ protected settlement: PromiseSettledResult<T> | null; /** Constructor accepts a normal Promise executor function like `new * Unpromise((resolve, reject) => {...})` or accepts a pre-existing Promise * like `new Unpromise(existingPromise)`. Adds `.then()` and `.catch()` * handlers to the Promise. These handlers pass fulfilment and rejection * notifications to downstream subscribers and maintains records of value * or error if the Promise ever settles. */ protected constructor(promise: Promise<T>); protected constructor(promise: PromiseLike<T>); protected constructor(executor: PromiseExecutor<T>); /** Create a promise that mitigates uncontrolled subscription to a long-lived * Promise via .then() and .catch() - otherwise a source of memory leaks. * * The returned promise has an `unsubscribe()` method which can be called when * the Promise is no longer being tracked by application logic, and which * ensures that there is no reference chain from the original promise to the * new one, and therefore no memory leak. * * If original promise has not yet settled, this adds a new unique promise * that listens to then/catch events, along with an `unsubscribe()` method to * detach it. * * If original promise has settled, then creates a new Promise.resolve() or * Promise.reject() and provided unsubscribe is a noop. * * If you call `unsubscribe()` before the returned Promise has settled, it * will never settle. */ subscribe(): SubscribedPromise<T>; /** STANDARD PROMISE METHODS (but returning a SubscribedPromise) */ then<TResult1 = T, TResult2 = never>(onfulfilled?: ((value: T) => TResult1 | PromiseLike<TResult1>) | null, onrejected?: ((reason: any) => TResult2 | PromiseLike<TResult2>) | null): SubscribedPromise<TResult1 | TResult2>; catch<TResult = never>(onrejected?: ((reason: any) => TResult | PromiseLike<TResult>) | null): SubscribedPromise<T | TResult>; finally(onfinally?: (() => void) | null): SubscribedPromise<T>; /** TOSTRING SUPPORT */ readonly [Symbol.toStringTag] = "Unpromise"; /** Unpromise STATIC METHODS */ /** Create or Retrieve the proxy Unpromise (a re-used Unpromise for the VM lifetime * of the provided Promise reference) */ static proxy<T>(promise: PromiseLike<T>): ProxyPromise<T>; /** Create and store an Unpromise keyed by an original Promise. */ protected static createSubscribablePromise<T>(promise: PromiseLike<T>): Unpromise<T>; /** Retrieve a previously-created Unpromise keyed by an original Promise. */ protected static getSubscribablePromise<T>(promise: PromiseLike<T>): ProxyPromise<T> | undefined; /** Promise STATIC METHODS */ /** Lookup the Unpromise for this promise, and derive a SubscribedPromise from * it (that can be later unsubscribed to eliminate Memory leaks) */ static resolve<T>(value: T | PromiseLike<T>): SubscribedPromise<Awaited<T>>; /** Perform Promise.any() via SubscribedPromises, then unsubscribe them. * Equivalent to Promise.any but eliminates memory leaks from long-lived * promises accumulating .then() and .catch() subscribers. */ static any<T extends readonly unknown[] | []>(values: T): Promise<Awaited<T[number]>>; /** Perform Promise.race via SubscribedPromises, then unsubscribe them. * Equivalent to Promise.race but eliminates memory leaks from long-lived * promises accumulating .then() and .catch() subscribers. */ static race<T extends readonly unknown[] | []>(values: T): Promise<Awaited<T[number]>>; /** Create a race of SubscribedPromises that will fulfil to a single winning * Promise (in a 1-Tuple). Eliminates memory leaks from long-lived promises * accumulating .then() and .catch() subscribers. Allows simple logic to * consume the result, like... * ```ts * const [ winner ] = await Unpromise.race([ promiseA, promiseB ]); * if(winner === promiseB){ * const result = await promiseB; * // do the thing * } * ``` * */ static raceReferences<TPromise extends Promise<unknown>>(promises: readonly TPromise[]): Promise<readonly [TPromise]>; } /** Promises a 1-tuple containing the original promise when it resolves. Allows * awaiting the eventual Promise ***reference*** (easy to destructure and * exactly compare with ===). Avoids resolving to the Promise ***value*** (which * may be ambiguous and therefore hard to identify as the winner of a race). * You can call unsubscribe on the Promise to mitigate memory leaks. * */ export declare function resolveSelfTuple<TPromise extends Promise<unknown>>(promise: TPromise): SubscribedPromise<readonly [TPromise]>; //# sourceMappingURL=unpromise.d.ts.map