-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy path553-deep-object-to-unique.ts
73 lines (64 loc) · 2.21 KB
/
553-deep-object-to-unique.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
/**
* 553 - Deep object to unique
*
* TypeScript has structural type system, but sometimes you want a function to accept only some previously well-defined
* unique objects (as in the nominal type system), and not any objects that have the required fields.
*
* Create a type that takes an object and makes it and all deeply nested objects in it unique,
* while preserving the string and numeric keys of all objects, and the values of all properties on these keys.
*
* The original type and the resulting unique type must be mutually assignable, but not identical.
*
* For example,
*
* ```ts
* import { Equal } from "@type-challenges/utils"
*
* type Foo = { foo: 2; bar: { 0: 1 }; baz: { 0: 1 } }
*
* type UniqFoo = DeepObjectToUniq<Foo>
*
* declare let foo: Foo
* declare let uniqFoo: UniqFoo
*
* uniqFoo = foo // ok
* foo = uniqFoo // ok
*
* type T0 = Equal<UniqFoo, Foo> // false
* type T1 = UniqFoo["foo"] // 2
* type T2 = Equal<UniqFoo["bar"], UniqFoo["baz"]> // false
* type T3 = UniqFoo["bar"][0] // 1
* type T4 = Equal<keyof Foo & string, keyof UniqFoo & string> // true
* ```
*
*
*/
/* _____________ Your Code Here _____________ */
type DeepObjectToUniq<O extends object, F extends unknown[] = []> = Omit<{
[K in keyof O]: O[K] extends object
? DeepObjectToUniq<O[K], [...F, [K, O]]>
: O[K]
} & { readonly [K in symbol]?: F }, never>
/* _____________ Test Cases _____________ */
import type { Equal, IsFalse, IsTrue } from '@type-challenges/utils'
type Quz = { quz: 4 }
type Foo = { foo: 2, baz: Quz, bar: Quz }
type Bar = { foo: 2, baz: Quz, bar: Quz & { quzz?: 0 } }
type UniqQuz = DeepObjectToUniq<Quz>
type UniqFoo = DeepObjectToUniq<Foo>
type UniqBar = DeepObjectToUniq<Bar>
declare let foo: Foo
declare let uniqFoo: UniqFoo
uniqFoo = foo
foo = uniqFoo
type cases = [
IsFalse<Equal<UniqQuz, Quz>>,
IsFalse<Equal<UniqFoo, Foo>>,
IsTrue<Equal<UniqFoo['foo'], Foo['foo']>>,
IsTrue<Equal<UniqFoo['bar']['quz'], Foo['bar']['quz']>>,
IsFalse<Equal<UniqQuz, UniqFoo['baz']>>,
IsFalse<Equal<UniqFoo['bar'], UniqFoo['baz']>>,
IsFalse<Equal<UniqBar['baz'], UniqFoo['baz']>>,
IsTrue<Equal<keyof UniqBar['baz'], keyof UniqFoo['baz']>>,
IsTrue<Equal<keyof Foo, keyof UniqFoo & string>>,
]