-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathdictionary.ts
94 lines (79 loc) · 2.63 KB
/
dictionary.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
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
import { objectRuntype } from './object'
import {
createFail,
InternalRuntypeOf,
isFail,
propagateFail,
setupInternalRuntype,
} from './runtype'
import type { Runtype } from './runtype'
import { debugValue } from './runtypeError'
const internalObjectRuntype: InternalRuntypeOf<typeof objectRuntype> = objectRuntype
function dictionaryRuntype<T extends string, U>(
keyRuntype: Runtype<T>,
valueRuntype: Runtype<U>,
) {
const keyRt: InternalRuntypeOf<typeof keyRuntype> = keyRuntype
const valueRt: InternalRuntypeOf<typeof valueRuntype> = valueRuntype
const isPure = !!(keyRt.meta?.isPure && valueRt.meta?.isPure)
return setupInternalRuntype<Record<T, U>>(
(v, failOrThrow) => {
const o = internalObjectRuntype(v, failOrThrow)
if (isFail(o)) {
return propagateFail(failOrThrow, o, v)
}
if (Object.getOwnPropertySymbols(o).length) {
return createFail(
failOrThrow,
`invalid key in dictionary: ${debugValue(
Object.getOwnPropertySymbols(o),
)}`,
v,
)
}
// optimize allocations: only create a copy if any of the key runtypes
// return a different object - otherwise return value as is
const res = (isPure ? o : {}) as { [key: string]: U }
for (const key in o) {
if (!Object.prototype.hasOwnProperty.call(o, key)) {
continue
}
if (key === '__proto__') {
// e.g. someone tried to sneak __proto__ into this object and that
// will cause havoc when assigning it to a new object (in case its impure)
return createFail(
failOrThrow,
`invalid key in dictionary: ${debugValue(key)}`,
v,
)
}
const keyOrFail = keyRt(key, failOrThrow)
if (isFail(keyOrFail)) {
return propagateFail(failOrThrow, keyOrFail, v)
}
const value = o[key as keyof typeof o]
const valueOrFail = valueRt(value, failOrThrow)
if (isFail(valueOrFail)) {
return propagateFail(failOrThrow, valueOrFail, v)
}
if (!isPure) {
res[keyOrFail] = valueOrFail
}
}
return res
},
{ isPure },
)
}
/**
* An object that matches a Typecript `Record<KeyType, ValueType>` type.
*
* You pass a runtype for the objects keys and one for its values.
* Keeps you save from unwanted propertiers and evil __proto__ injections.
*/
export function dictionary<T extends Runtype<any>, U extends Runtype<any>>(
keyRuntype: T,
valueRuntype: U,
): Runtype<Record<ReturnType<T>, ReturnType<U>>> {
return dictionaryRuntype(keyRuntype, valueRuntype)
}