UNPKG

ts-results-es

Version:

A TypeScript implementation of Rust's Result and Option objects.

159 lines (149 loc) 5.71 kB
import { AsyncResult } from './asyncresult.js'; import { Option, Some } from './option.js'; /** * An async-aware `Option` counterpart. * * Can be combined with asynchronous code without having to ``await`` anything right until * the moment when you're ready to extract the final ``Option`` out of it. * * Can also be combined with synchronous code for convenience. */ export class AsyncOption<T> { /** * A promise that resolves to a synchronous ``Option``. * * You can await it to convert `AsyncOption<T>` to `Option<T>`, but prefer * awaiting `AsyncOption` directly (see: `then()`). Only use this property * if you need the underlying Promise for specific use cases. */ promise: Promise<Option<T>>; /** * Constructs an `AsyncOption` from an `Option` or a `Promise` of an `Option`. * * @example * ```typescript * const option = new AsyncOption(Promise.resolve(Some('username'))) * ``` */ constructor(start: Option<T> | Promise<Option<T>>) { this.promise = Promise.resolve(start); } /** * Calls `mapper` if the option is `Some`, otherwise keeps the `None` value intact. * This function can be used for control flow based on `Option` values. * * @example * ```typescript * let hasValue = Some(1).toAsyncOption() * let noValue = None.toAsyncOption() * * await hasValue.andThen(async (value) => Some(value * 2)).promise // Some(2) * await hasValue.andThen(async (value) => None).promise // None * await noValue.andThen(async (value) => Some(value * 2)).promise // None * ``` */ andThen<T2>(mapper: (val: T) => Option<T2> | Promise<Option<T2>> | AsyncOption<T2>): AsyncOption<T2> { return this.thenInternal(async (option) => { if (option.isNone()) { return option; } const mapped = mapper(option.value); return mapped instanceof AsyncOption ? mapped.promise : mapped; }); } /** * Maps an `AsyncOption<T>` to `AsyncOption<U>` by applying a function to a contained * `Some` value, leaving a `None` value untouched. * * This function can be used to compose the results of two functions. * * @example * ```typescript * let hasValue = Some(1).toAsyncOption() * let noValue = None.toAsyncOption() * * await hasValue.map(async (value) => value * 2).promise // Some(2) * await noValue.map(async (value) => value * 2).promise // None * ``` */ map<U>(mapper: (val: T) => U | Promise<U>): AsyncOption<U> { return this.thenInternal(async (option) => { if (option.isNone()) { return option; } return Some(await mapper(option.value)); }); } /** * Returns the value from `other` if this `AsyncOption` contains `None`, otherwise returns self. * * If `other` is a result of a function call consider using `orElse` instead, it will * only evaluate the function when needed. * * @example * ``` * const noValue = new AsyncOption(None) * const hasValue = new AsyncOption(Some(1)) * * await noValue.or(Some(123)).promise // Some(123) * await hasValue.or(Some(123)).promise // Some(1) * ``` */ or<U>(other: Option<U> | AsyncOption<U> | Promise<Option<U>>): AsyncOption<T | U> { return this.orElse(() => other); } /** * Returns the value obtained by calling `other` if this `AsyncOption` contains `None`, otherwise * returns self. * * @example * ``` * const noValue = new AsyncOption(None) * const hasValue = new AsyncOption(Some(1)) * * await noValue.orElse(() => Some(123)).promise // Some(123) * await hasValue.orElse(() => Some(123)).promise // Some(1) * ``` */ orElse<U>(other: () => Option<U> | AsyncOption<U> | Promise<Option<U>>): AsyncOption<T | U> { return this.thenInternal(async (option): Promise<Option<T | U>> => { if (option.isSome()) { return option; } const otherValue = other(); return otherValue instanceof AsyncOption ? otherValue.promise : otherValue; }); } /** * Converts an `AsyncOption<T>` to an `AsyncResult<T, E>` so that `None` is converted to * `Err(error)` and `Some` is converted to `Ok`. */ toResult<E>(error: E): AsyncResult<T, E> { return new AsyncResult(this.promise.then((option) => option.toResult(error))); } /** * Makes `AsyncOption` awaitable by implementing the thenable interface. * This allows you to use `await` directly on `AsyncOption` instances. * * See the [Promise.then() documentation](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Promise/then) * for details on the thenable interface. * * @example * ```typescript * const asyncOption = new AsyncOption(Some(42)) * const option = await asyncOption // Returns Option<number> * * // Equivalent to: * const option2 = await asyncOption.promise * ``` */ then<TResult1 = Option<T>, TResult2 = never>( onfulfilled?: ((value: Option<T>) => TResult1 | PromiseLike<TResult1>) | null, onrejected?: ((reason: any) => TResult2 | PromiseLike<TResult2>) | null, ): Promise<TResult1 | TResult2> { return this.promise.then(onfulfilled, onrejected); } private thenInternal<T2>(mapper: (option: Option<T>) => Promise<Option<T2>>): AsyncOption<T2> { return new AsyncOption(this.promise.then(mapper)); } }