UNPKG

ts-pattern

Version:

The exhaustive Pattern Matching library for TypeScript.

162 lines (161 loc) 8.37 kB
import type * as symbols from '../internals/symbols.js'; import type { Pattern, MatchedValue } from './Pattern.js'; import type { InvertPatternForExclude, InvertPattern } from './InvertPattern.js'; import type { DeepExclude } from './DeepExclude.js'; import type { Union, GuardValue, IsNever } from './helpers.js'; import type { FindSelected } from './FindSelected.js'; export type PickReturnValue<a, b> = a extends symbols.unset ? b : a; interface NonExhaustiveError<i> { __nonExhaustive: never; } interface TSPatternError<i> { __nonExhaustive: never; } /** * #### Match * An interface to create a pattern matching clause. */ export type Match<i, o, handledCases extends any[] = [], inferredOutput = never> = { /** * `.with(pattern, handler)` Registers a pattern and an handler function that * will be called if the pattern matches the input value. * * [Read the documentation for `.with()` on GitHub](https://github.com/gvergnaud/ts-pattern#with) **/ with<const p extends Pattern<i>, c, value extends MatchedValue<i, InvertPattern<p, i>>>(pattern: IsNever<p> extends true ? Pattern<i> : p, handler: (selections: FindSelected<value, p>, value: value) => PickReturnValue<o, c>): InvertPatternForExclude<p, value> extends infer excluded ? Match<Exclude<i, excluded>, o, [ ...handledCases, excluded ], Union<inferredOutput, c>> : never; with<const p1 extends Pattern<i>, const p2 extends Pattern<i>, c, p extends p1 | p2, value extends p extends any ? MatchedValue<i, InvertPattern<p, i>> : never>(p1: p1, p2: p2, handler: (value: value) => PickReturnValue<o, c>): [ InvertPatternForExclude<p1, value>, InvertPatternForExclude<p2, value> ] extends [infer excluded1, infer excluded2] ? Match<Exclude<i, excluded1 | excluded2>, o, [ ...handledCases, excluded1, excluded2 ], Union<inferredOutput, c>> : never; with<const p1 extends Pattern<i>, const p2 extends Pattern<i>, const p3 extends Pattern<i>, const ps extends readonly Pattern<i>[], c, p extends p1 | p2 | p3 | ps[number], value extends MatchedValue<i, InvertPattern<p, i>>>(...args: [ p1: p1, p2: p2, p3: p3, ...patterns: ps, handler: (value: value) => PickReturnValue<o, c> ]): [ InvertPatternForExclude<p1, value>, InvertPatternForExclude<p2, value>, InvertPatternForExclude<p3, value>, MakeTuples<ps, value> ] extends [ infer excluded1, infer excluded2, infer excluded3, infer excludedRest ] ? Match<Exclude<i, excluded1 | excluded2 | excluded3 | Extract<excludedRest, any[]>[number]>, o, [ ...handledCases, excluded1, excluded2, excluded3, ...Extract<excludedRest, any[]> ], Union<inferredOutput, c>> : never; with<const pat extends Pattern<i>, pred extends (value: MatchedValue<i, InvertPattern<pat, i>>) => unknown, c, value extends GuardValue<pred>>(pattern: pat, predicate: pred, handler: (selections: FindSelected<value, pat>, value: value) => PickReturnValue<o, c>): pred extends (value: any) => value is infer narrowed ? Match<Exclude<i, narrowed>, o, [ ...handledCases, narrowed ], Union<inferredOutput, c>> : Match<i, o, handledCases, Union<inferredOutput, c>>; /** * `.when(predicate, handler)` Registers a predicate function and an handler function. * If the predicate returns true, the handler function will be called. * * [Read the documentation for `.when()` on GitHub](https://github.com/gvergnaud/ts-pattern#when) **/ when<pred extends (value: i) => unknown, c, value extends GuardValue<pred>>(predicate: pred, handler: (value: value) => PickReturnValue<o, c>): pred extends (value: any) => value is infer narrowed ? Match<Exclude<i, narrowed>, o, [ ...handledCases, narrowed ], Union<inferredOutput, c>> : Match<i, o, handledCases, Union<inferredOutput, c>>; /** * `.otherwise()` takes a **default handler function** that will be * called if no previous pattern matched your input. * * Equivalent to `.with(P._, () => x).exhaustive()` * * [Read the documentation for `.otherwise()` on GitHub](https://github.com/gvergnaud/ts-pattern#otherwise) * **/ otherwise<c>(handler: (value: i) => PickReturnValue<o, c>): PickReturnValue<o, Union<inferredOutput, c>>; /** * `.exhaustive()` checks that all cases are handled, and returns the result value. * * If you get a `NonExhaustiveError`, it means that you aren't handling * all cases. You should probably add another `.with(...)` clause * to match the missing case and prevent runtime errors. * * [Read the documentation for `.exhaustive()` on GitHub](https://github.com/gvergnaud/ts-pattern#exhaustive) * */ exhaustive: DeepExcludeAll<i, handledCases> extends infer remainingCases ? [remainingCases] extends [never] ? Exhaustive<o, inferredOutput> : NonExhaustiveError<remainingCases> : never; /** * `.run()` return the resulting value. * * ⚠️ calling this function is unsafe, and may throw if no pattern matches your input. */ run(): PickReturnValue<o, inferredOutput>; /** * `.returnType<T>()` Lets you specify the return type for all of your branches. * * [Read the documentation for `.returnType()` on GitHub](https://github.com/gvergnaud/ts-pattern#returnType) */ returnType: [inferredOutput] extends [never] ? <output>() => Match<i, output, handledCases> : TSPatternError<'calling `.returnType<T>()` is only allowed directly after `match(...)`.'>; }; /** * Potential for optimization here: * * Since DeepExclude distributes the union of the input type, it can * generate very large union types on patterns touching several unions at once. * If we were sorting patterns from those which distribute the smallest * amount of union types to those which distribute the largest, we would eliminate * cheap cases more quickly and have less cases in the input type for patterns * that will be expensive to exclude. * * This pre supposes that we have a cheap way of telling if the number * of union types a pattern touches and a cheap way of sorting the tuple * of patterns. * - For the first part, we could reuse `FindMatchingUnions` and pick the `length` * of the returned tuple. * - For the second part though I'm not aware a cheap way of sorting a tuple. */ type DeepExcludeAll<a, tupleList extends any[]> = [a] extends [never] ? never : tupleList extends [infer excluded, ...infer tail] ? DeepExcludeAll<DeepExclude<a, excluded>, tail> : a; type MakeTuples<ps extends readonly any[], value> = { -readonly [index in keyof ps]: InvertPatternForExclude<ps[index], value>; }; /** * The type of an overloaded function for `.exhaustive`, * permitting calling it with or without a catch-all handler function. * * By default, TS-Pattern will throw an error if a runtime value isn't handled. */ type Exhaustive<output, inferredOutput> = { /** * `.exhaustive()` checks that all cases are handled, and returns the result value. * * If you get a `NonExhaustiveError`, it means that you aren't handling * all cases. You should probably add another `.with(...)` clause * to match the missing case and prevent runtime errors. * * [Read the documentation for `.exhaustive()` on GitHub](https://github.com/gvergnaud/ts-pattern#exhaustive) * */ (): PickReturnValue<output, inferredOutput>; /** * `.exhaustive(fallback)` checks that all cases are handled and returns the result value. * * The fallback function will be called if your input value doesn't match any pattern. * This can only happen if the value you passed to `match` has an incorrect type. * * If you get a `NonExhaustiveError`, it means that you aren't handling * all cases. You should probably add another `.with(...)` clause * to match the missing case. * * [Read the documentation for `.exhaustive()` on GitHub](https://github.com/gvergnaud/ts-pattern#exhaustive) */ <otherOutput>(handler: (unexpectedValue: unknown) => PickReturnValue<output, otherOutput>): PickReturnValue<output, Union<inferredOutput, otherOutput>>; }; export {};