UNPKG

pragmatic-fp-ts

Version:

Opinionated functional programming library with easy use in mind

33 lines (30 loc) 1.38 kB
type Lookup<T, K extends keyof any, Else = never> = K extends keyof T ? T[K] : Else; type Tail<T extends any[]> = ((...t: T) => void) extends (x: any, ...u: infer U) => void ? U : never; type Func1 = (arg: any) => any; type ArgType<F, Else = never> = F extends (arg: infer A) => any ? A : Else; type AsChain<F extends [Func1, ...Func1[]], G extends Func1[] = Tail<F>> = { [K in keyof F]: (arg: ArgType<F[K]>) => ArgType<Lookup<G, K, any>, any>; }; type LastIndexOf<T extends any[]> = ((...x: T) => void) extends (y: any, ...z: infer U) => void ? U["length"] : never; export function flow<F extends [(arg: any) => any, ...Array<(arg: any) => any>]>( ...f: F & AsChain<F> ): (arg: ArgType<F[0]>) => ReturnType<F[LastIndexOf<F>]>; /** * Composes an arbitrary number of functions into a new function, the * result being a function piping their results as argument into the * next function, with data flow from left to right. The first * function may be n-ary while all others must be unary * * Note: Generic function types cannot be inferred, resulting in a * TypeScript error. Successive (any) -> any functions might break * type inference. * * (((A1, B, C,...) -> A2) -> (A2 -> A3) ... -> (An-1 -> An)) -> (A1, B, C,...) -> An */ export function flow(...fns: any) { return (startValue: any) => fns.reduce((val: any, fn: any) => fn(val), startValue); }