|
| 1 | +//// [tests/cases/compiler/mappedTypeRemappingModifierMerging.ts] //// |
| 2 | + |
| 3 | +=== mappedTypeRemappingModifierMerging.ts === |
| 4 | +// Mapped types with key remapping should merge modifiers consistently |
| 5 | +// when multiple keys map to the same output key |
| 6 | + |
| 7 | +type RemapKeyToInitialPart<T> = { |
| 8 | +>RemapKeyToInitialPart : Symbol(RemapKeyToInitialPart, Decl(mappedTypeRemappingModifierMerging.ts, 0, 0)) |
| 9 | +>T : Symbol(T, Decl(mappedTypeRemappingModifierMerging.ts, 3, 27)) |
| 10 | + |
| 11 | + [K in keyof T as K extends `${infer First}.${infer _Rest}` ? First : K]: null; |
| 12 | +>K : Symbol(K, Decl(mappedTypeRemappingModifierMerging.ts, 4, 5)) |
| 13 | +>T : Symbol(T, Decl(mappedTypeRemappingModifierMerging.ts, 3, 27)) |
| 14 | +>K : Symbol(K, Decl(mappedTypeRemappingModifierMerging.ts, 4, 5)) |
| 15 | +>First : Symbol(First, Decl(mappedTypeRemappingModifierMerging.ts, 4, 39)) |
| 16 | +>_Rest : Symbol(_Rest, Decl(mappedTypeRemappingModifierMerging.ts, 4, 54)) |
| 17 | +>First : Symbol(First, Decl(mappedTypeRemappingModifierMerging.ts, 4, 39)) |
| 18 | +>K : Symbol(K, Decl(mappedTypeRemappingModifierMerging.ts, 4, 5)) |
| 19 | + |
| 20 | +}; |
| 21 | + |
| 22 | +// Both should produce { foo?: null } since at least one input is optional |
| 23 | +type FirstOptional = RemapKeyToInitialPart<{ |
| 24 | +>FirstOptional : Symbol(FirstOptional, Decl(mappedTypeRemappingModifierMerging.ts, 5, 2)) |
| 25 | +>RemapKeyToInitialPart : Symbol(RemapKeyToInitialPart, Decl(mappedTypeRemappingModifierMerging.ts, 0, 0)) |
| 26 | + |
| 27 | + "foo.bar"?: string; |
| 28 | +>"foo.bar" : Symbol("foo.bar", Decl(mappedTypeRemappingModifierMerging.ts, 8, 44)) |
| 29 | + |
| 30 | + "foo.baz": number; |
| 31 | +>"foo.baz" : Symbol("foo.baz", Decl(mappedTypeRemappingModifierMerging.ts, 9, 23)) |
| 32 | + |
| 33 | +}>; |
| 34 | + |
| 35 | +type FirstRequired = RemapKeyToInitialPart<{ |
| 36 | +>FirstRequired : Symbol(FirstRequired, Decl(mappedTypeRemappingModifierMerging.ts, 11, 3)) |
| 37 | +>RemapKeyToInitialPart : Symbol(RemapKeyToInitialPart, Decl(mappedTypeRemappingModifierMerging.ts, 0, 0)) |
| 38 | + |
| 39 | + "foo.baz": number; |
| 40 | +>"foo.baz" : Symbol("foo.baz", Decl(mappedTypeRemappingModifierMerging.ts, 13, 44)) |
| 41 | + |
| 42 | + "foo.bar"?: string; |
| 43 | +>"foo.bar" : Symbol("foo.bar", Decl(mappedTypeRemappingModifierMerging.ts, 14, 22)) |
| 44 | + |
| 45 | +}>; |
| 46 | + |
| 47 | +// Test that they are equivalent |
| 48 | +const testOptional: FirstOptional = { foo: null }; |
| 49 | +>testOptional : Symbol(testOptional, Decl(mappedTypeRemappingModifierMerging.ts, 19, 5)) |
| 50 | +>FirstOptional : Symbol(FirstOptional, Decl(mappedTypeRemappingModifierMerging.ts, 5, 2)) |
| 51 | +>foo : Symbol(foo, Decl(mappedTypeRemappingModifierMerging.ts, 19, 37)) |
| 52 | + |
| 53 | +const testOptional2: FirstOptional = {}; |
| 54 | +>testOptional2 : Symbol(testOptional2, Decl(mappedTypeRemappingModifierMerging.ts, 20, 5)) |
| 55 | +>FirstOptional : Symbol(FirstOptional, Decl(mappedTypeRemappingModifierMerging.ts, 5, 2)) |
| 56 | + |
| 57 | +const testRequired: FirstRequired = { foo: null }; |
| 58 | +>testRequired : Symbol(testRequired, Decl(mappedTypeRemappingModifierMerging.ts, 22, 5)) |
| 59 | +>FirstRequired : Symbol(FirstRequired, Decl(mappedTypeRemappingModifierMerging.ts, 11, 3)) |
| 60 | +>foo : Symbol(foo, Decl(mappedTypeRemappingModifierMerging.ts, 22, 37)) |
| 61 | + |
| 62 | +const testRequired2: FirstRequired = {}; |
| 63 | +>testRequired2 : Symbol(testRequired2, Decl(mappedTypeRemappingModifierMerging.ts, 23, 5)) |
| 64 | +>FirstRequired : Symbol(FirstRequired, Decl(mappedTypeRemappingModifierMerging.ts, 11, 3)) |
| 65 | + |
| 66 | +// Readonly should work the same way |
| 67 | +type RemapWithReadonly<T> = { |
| 68 | +>RemapWithReadonly : Symbol(RemapWithReadonly, Decl(mappedTypeRemappingModifierMerging.ts, 23, 40)) |
| 69 | +>T : Symbol(T, Decl(mappedTypeRemappingModifierMerging.ts, 26, 23)) |
| 70 | + |
| 71 | + [K in keyof T as K extends `${infer First}.${string}` ? First : K]: null; |
| 72 | +>K : Symbol(K, Decl(mappedTypeRemappingModifierMerging.ts, 27, 5)) |
| 73 | +>T : Symbol(T, Decl(mappedTypeRemappingModifierMerging.ts, 26, 23)) |
| 74 | +>K : Symbol(K, Decl(mappedTypeRemappingModifierMerging.ts, 27, 5)) |
| 75 | +>First : Symbol(First, Decl(mappedTypeRemappingModifierMerging.ts, 27, 39)) |
| 76 | +>First : Symbol(First, Decl(mappedTypeRemappingModifierMerging.ts, 27, 39)) |
| 77 | +>K : Symbol(K, Decl(mappedTypeRemappingModifierMerging.ts, 27, 5)) |
| 78 | + |
| 79 | +}; |
| 80 | + |
| 81 | +type FirstReadonly = RemapWithReadonly<{ |
| 82 | +>FirstReadonly : Symbol(FirstReadonly, Decl(mappedTypeRemappingModifierMerging.ts, 28, 2)) |
| 83 | +>RemapWithReadonly : Symbol(RemapWithReadonly, Decl(mappedTypeRemappingModifierMerging.ts, 23, 40)) |
| 84 | + |
| 85 | + readonly "foo.bar": string; |
| 86 | +>"foo.bar" : Symbol("foo.bar", Decl(mappedTypeRemappingModifierMerging.ts, 30, 40)) |
| 87 | + |
| 88 | + "foo.baz": number; |
| 89 | +>"foo.baz" : Symbol("foo.baz", Decl(mappedTypeRemappingModifierMerging.ts, 31, 31)) |
| 90 | + |
| 91 | +}>; |
| 92 | + |
| 93 | +type SecondReadonly = RemapWithReadonly<{ |
| 94 | +>SecondReadonly : Symbol(SecondReadonly, Decl(mappedTypeRemappingModifierMerging.ts, 33, 3)) |
| 95 | +>RemapWithReadonly : Symbol(RemapWithReadonly, Decl(mappedTypeRemappingModifierMerging.ts, 23, 40)) |
| 96 | + |
| 97 | + "foo.baz": number; |
| 98 | +>"foo.baz" : Symbol("foo.baz", Decl(mappedTypeRemappingModifierMerging.ts, 35, 41)) |
| 99 | + |
| 100 | + readonly "foo.bar": string; |
| 101 | +>"foo.bar" : Symbol("foo.bar", Decl(mappedTypeRemappingModifierMerging.ts, 36, 22)) |
| 102 | + |
| 103 | +}>; |
| 104 | + |
| 105 | +declare const ro1: FirstReadonly; |
| 106 | +>ro1 : Symbol(ro1, Decl(mappedTypeRemappingModifierMerging.ts, 40, 13)) |
| 107 | +>FirstReadonly : Symbol(FirstReadonly, Decl(mappedTypeRemappingModifierMerging.ts, 28, 2)) |
| 108 | + |
| 109 | +declare const ro2: SecondReadonly; |
| 110 | +>ro2 : Symbol(ro2, Decl(mappedTypeRemappingModifierMerging.ts, 41, 13)) |
| 111 | +>SecondReadonly : Symbol(SecondReadonly, Decl(mappedTypeRemappingModifierMerging.ts, 33, 3)) |
| 112 | + |
| 113 | +// Both should be readonly |
| 114 | +ro1.foo = null; // Error |
| 115 | +>ro1.foo : Symbol(foo) |
| 116 | +>ro1 : Symbol(ro1, Decl(mappedTypeRemappingModifierMerging.ts, 40, 13)) |
| 117 | +>foo : Symbol(foo) |
| 118 | + |
| 119 | +ro2.foo = null; // Error |
| 120 | +>ro2.foo : Symbol(foo) |
| 121 | +>ro2 : Symbol(ro2, Decl(mappedTypeRemappingModifierMerging.ts, 41, 13)) |
| 122 | +>foo : Symbol(foo) |
| 123 | + |
0 commit comments