-
Notifications
You must be signed in to change notification settings - Fork 15
/
Copy pathget.js
101 lines (97 loc) · 3.23 KB
/
get.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
88
89
90
91
92
93
94
95
96
97
98
99
100
101
const isPromise = require('./_internal/isPromise')
const __ = require('./_internal/placeholder')
const curry3 = require('./_internal/curry3')
const isArray = require('./_internal/isArray')
const isObject = require('./_internal/isObject')
const getByPath = require('./_internal/getByPath')
// _get(object Object, path string, defaultValue function|any)
const _get = function (object, path, defaultValue) {
const result = object == null ? undefined : getByPath(object, path)
return result === undefined
? typeof defaultValue == 'function' ? defaultValue(object) : defaultValue
: result
}
/**
* @name get
*
* @synopsis
* ```coffeescript [specscript]
* get(
* object Promise|Object,
* path string|number|Array<string|number>,
* defaultValue? function|any
* ) -> result Promise|Object
*
* get(
* path string|number|Array<string|number>,
* defaultValue? function|any
* )(object Object) -> result Promise|Object
* ```
*
* @description
* Accesses a property of an object given a path denoted by a string, number, or an array of string or numbers.
*
* ```javascript [playground]
* const obj = { hello: 'world' }
*
* console.log(get(obj, 'hello')) // world
* ```
*
* `get` supports a lazy API for composability
*
* ```javascript [playground]
* const obj = { hello: 'world' }
*
* const getHello = get('hello')
*
* console.log(getHello({ hello: 'world' })) // world
* ```
*
* If the value at the end of the path is not found on the object, returns an optional default value. The default value can be a function resolver that takes the object as an argument. If no default value is provided, returns `undefined`. The function resolver may be asynchronous (returns a promise).
*
* ```javascript [playground]
* const getHelloWithDefaultValue = get('hello', 'default')
*
* console.log(getHelloWithDefaultValue({ foo: 'bar' })) // default
*
* const getHelloWithDefaultResolver = get('hello', object => object.foo)
*
* console.log(getHelloWithDefaultResolver({ foo: 'bar' })) // bar
* ```
*
* `get` supports three types of path patterns for nested property access.
*
* * dot delimited - `'a.b.c'`
* * bracket notation - `'a[0].value'`
* * an array of keys or indices - `['a', 0, 'value']`
*
* ```javascript [playground]
* const getABC0 = get('a.b.c[0]')
*
* console.log(getABC0({ a: { b: { c: ['hello'] } } })) // hello
*
* const get00000DotNotation = get('0.0.0.0.0')
* const get00000BracketNotation = get('[0][0][0][0][0]')
* const get00000ArrayNotation = get([0, 0, 0, 0, 0])
*
* console.log(get00000DotNotation([[[[['foo']]]]])) // foo
* console.log(get00000BracketNotation([[[[['foo']]]]])) // foo
* console.log(get00000ArrayNotation([[[[['foo']]]]])) // foo
* ```
*
* Any promises passed in argument position are resolved for their values before further execution. This only applies to the eager version of the API.
*
* ```javascript [playground]
* get(Promise.resolve({ a: 1 }), 'a').then(console.log) // 1
* ```
*/
const get = function (arg0, arg1, arg2) {
if (typeof arg0 == 'string' || typeof arg0 == 'number' || isArray(arg0)) {
return curry3(_get, __, arg0, arg1)
}
if (isPromise(arg0)) {
return arg0.then(curry3(_get, __, arg1, arg2))
}
return _get(arg0, arg1, arg2)
}
module.exports = get