-
Notifications
You must be signed in to change notification settings - Fork 28
/
Copy pathnullishChain.ts
52 lines (50 loc) · 1.93 KB
/
nullishChain.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
import type { PromType, λ } from "./types.ts";
/**
* Given a list of functions that accept the same parameters, returns a function
* that given these arguments returns the result of the first function whose
* result is not nullish.
*
* This is equivalent to chaining together invocations of the passed in
* functions with the given arguments with nullish coalescing _(`??`)_ operators.
*
* @example
* ```
* const isAdult = (age: number) => { if (n >= 18) return 'adult' }
* const isToddler = (age: number) => { if (n <= 3) return 'toddler' }
*
* const ageGroup = nullishChain(isAdult, isToddler, () => 'child')
*
* // this is functionally equivalent to:
* const ageGroup = age => isAdult(age) ?? isToddler(age) ?? 'child'
*
* ageGroup(1) // prints: 'toddler'
* ageGroup(10) // prints: 'child'
* ageGroup(50) // prints: 'adult'
* ```
*/
export const nullishChain = <FF extends λ, FR extends λ<Parameters<FF>>[]>(
...[fun, ...rest]: [FF, ...FR] | []
) =>
(...args: Parameters<FF>): ReturnType<FF | FR[number]> | undefined =>
!fun ? undefined : fun(...args) ?? nullishChain(...(rest as any))(...args);
/**
* Same as {@link nullishChain} but accept asynchronous functions too.
*
* @example
* ```
* const readFromCache = (id: string): Resource => { if (id in cache) return cache[id] }
* const readFromFile = (id: string): Resource => { if (fileExists(id)) return readFile(id) }
* const fetchFromNet = async (id: string): Promise<Resource> => await fetch(`someURL/${id}`)
*
* // async (id: string) => Promise<Resource>
* const getResource = asyncNullishChain(readFromCache, readFromFile, fetchFromNet)
* ```
*/
export const asyncNullishChain = <FF extends λ, FR extends λ<Parameters<FF>>[]>(
...[fun, ...rest]: [FF, ...FR] | []
) =>
async (
...args: Parameters<FF>
): Promise<PromType<ReturnType<FF | FR[number]>> | undefined> =>
!fun ? undefined : (await fun(...args)) ??
(await asyncNullishChain(...(rest as any))(...args));