This repository has been archived by the owner on Jan 10, 2024. It is now read-only.
/
idx.js
87 lines (83 loc) · 2.49 KB
/
idx.js
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
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
/**
* Copyright (c) Facebook, Inc. and its affiliates.
*
* This source code is licensed under the MIT license found in the
* LICENSE file in the root directory of this source tree.
*
* @providesModule idx
* @flow strict
* @format
*/
'use strict'; // eslint-disable-line strict
/**
* Traverses properties on objects and arrays. If an intermediate property is
* either null or undefined, it is instead returned. The purpose of this method
* is to simplify extracting properties from a chain of maybe-typed properties.
*
* === EXAMPLE ===
*
* Consider the following type:
*
* const props: {
* user: ?{
* name: string,
* friends: ?Array<User>,
* }
* };
*
* Getting to the friends of my first friend would resemble:
*
* props.user &&
* props.user.friends &&
* props.user.friends[0] &&
* props.user.friends[0].friends
*
* Instead, `idx` allows us to safely write:
*
* idx(props, _ => _.user.friends[0].friends)
*
* The second argument must be a function that returns one or more nested member
* expressions. Any other expression has undefined behavior.
*
* === NOTE ===
*
* The code below exists for the purpose of illustrating expected behavior and
* is not meant to be executed. The `idx` function is used in conjunction with a
* Babel transform that replaces it with better performing code:
*
* props.user == null ? props.user :
* props.user.friends == null ? props.user.friends :
* props.user.friends[0] == null ? props.user.friends[0] :
* props.user.friends[0].friends
*
* All this machinery exists due to the fact that an existential operator does
* not currently exist in JavaScript.
*/
function idx<Ti, Tv>(input: Ti, accessor: (input: Ti) => Tv): ?Tv {
try {
return accessor(input);
} catch (error) {
if (error instanceof TypeError) {
if (nullPattern.test(error.message)) {
return null;
} else if (undefinedPattern.test(error.message)) {
return undefined;
}
}
throw error;
}
}
/**
* Some actual error messages for null:
*
* TypeError: Cannot read property 'bar' of null
* TypeError: Cannot convert null value to object
* TypeError: foo is null
* TypeError: null has no properties
* TypeError: null is not an object (evaluating 'foo.bar')
* TypeError: null is not an object (evaluating '(" undefined ", null).bar')
*/
const nullPattern = /^null | null$|^[^(]* null /i;
const undefinedPattern = /^undefined | undefined$|^[^(]* undefined /i;
idx.default = idx;
module.exports = idx;