diff --git a/tests/baselines/reference/recursiveMappedTypes.errors.txt b/tests/baselines/reference/recursiveMappedTypes.errors.txt index c003204cd78..43a0e736d25 100644 --- a/tests/baselines/reference/recursiveMappedTypes.errors.txt +++ b/tests/baselines/reference/recursiveMappedTypes.errors.txt @@ -5,9 +5,10 @@ tests/cases/conformance/types/mapped/recursiveMappedTypes.ts(8,11): error TS2313 tests/cases/conformance/types/mapped/recursiveMappedTypes.ts(11,6): error TS2456: Type alias 'Recurse2' circularly references itself. tests/cases/conformance/types/mapped/recursiveMappedTypes.ts(12,11): error TS2313: Type parameter 'K' has a circular constraint. tests/cases/conformance/types/mapped/recursiveMappedTypes.ts(20,19): error TS2589: Type instantiation is excessively deep and possibly infinite. +tests/cases/conformance/types/mapped/recursiveMappedTypes.ts(66,25): error TS2313: Type parameter 'P' has a circular constraint. -==== tests/cases/conformance/types/mapped/recursiveMappedTypes.ts (7 errors) ==== +==== tests/cases/conformance/types/mapped/recursiveMappedTypes.ts (8 errors) ==== // Recursive mapped types simply appear empty type Recurse = { @@ -84,4 +85,25 @@ tests/cases/conformance/types/mapped/recursiveMappedTypes.ts(20,19): error TS258 type a = Remap1; // string[] type b = Remap2; // string[] + + // Repro from #29992 + + type NonOptionalKeys = { [P in keyof T]: undefined extends T[P] ? never : P }[keyof T]; + type Child = { [P in NonOptionalKeys]: T[P] } + ~~~~~~~~~~~~~~~~~~ +!!! error TS2313: Type parameter 'P' has a circular constraint. +!!! related TS2751 tests/cases/conformance/types/mapped/recursiveMappedTypes.ts:79:1: Circularity originates in type at this location. + + export interface ListWidget { + "type": "list", + "minimum_count": number, + "maximum_count": number, + "collapsable"?: boolean, //default to false, means all expanded + "each": Child; + } + + type ListChild = Child + + declare let x: ListChild; + x.type; \ No newline at end of file diff --git a/tests/baselines/reference/recursiveMappedTypes.js b/tests/baselines/reference/recursiveMappedTypes.js index e9e44a1df41..2cda37a8d45 100644 --- a/tests/baselines/reference/recursiveMappedTypes.js +++ b/tests/baselines/reference/recursiveMappedTypes.js @@ -60,6 +60,24 @@ type Remap2 = T extends object ? { [P in keyof T]: Remap2; } : T; type a = Remap1; // string[] type b = Remap2; // string[] + +// Repro from #29992 + +type NonOptionalKeys = { [P in keyof T]: undefined extends T[P] ? never : P }[keyof T]; +type Child = { [P in NonOptionalKeys]: T[P] } + +export interface ListWidget { + "type": "list", + "minimum_count": number, + "maximum_count": number, + "collapsable"?: boolean, //default to false, means all expanded + "each": Child; +} + +type ListChild = Child + +declare let x: ListChild; +x.type; //// [recursiveMappedTypes.js] @@ -70,9 +88,24 @@ function foo(arg) { return arg; } product.users; // (Transform | Transform)[] +x.type; //// [recursiveMappedTypes.d.ts] export declare type Circular = { [P in keyof T]: Circular; }; +declare type NonOptionalKeys = { + [P in keyof T]: undefined extends T[P] ? never : P; +}[keyof T]; +declare type Child = { + [P in NonOptionalKeys]: T[P]; +}; +export interface ListWidget { + "type": "list"; + "minimum_count": number; + "maximum_count": number; + "collapsable"?: boolean; + "each": Child; +} +export {}; diff --git a/tests/baselines/reference/recursiveMappedTypes.symbols b/tests/baselines/reference/recursiveMappedTypes.symbols index 1638dc03f56..777a6a722fa 100644 --- a/tests/baselines/reference/recursiveMappedTypes.symbols +++ b/tests/baselines/reference/recursiveMappedTypes.symbols @@ -165,3 +165,57 @@ type b = Remap2; // string[] >b : Symbol(b, Decl(recursiveMappedTypes.ts, 59, 26)) >Remap2 : Symbol(Remap2, Decl(recursiveMappedTypes.ts, 56, 51)) +// Repro from #29992 + +type NonOptionalKeys = { [P in keyof T]: undefined extends T[P] ? never : P }[keyof T]; +>NonOptionalKeys : Symbol(NonOptionalKeys, Decl(recursiveMappedTypes.ts, 60, 26)) +>T : Symbol(T, Decl(recursiveMappedTypes.ts, 64, 21)) +>P : Symbol(P, Decl(recursiveMappedTypes.ts, 64, 29)) +>T : Symbol(T, Decl(recursiveMappedTypes.ts, 64, 21)) +>T : Symbol(T, Decl(recursiveMappedTypes.ts, 64, 21)) +>P : Symbol(P, Decl(recursiveMappedTypes.ts, 64, 29)) +>P : Symbol(P, Decl(recursiveMappedTypes.ts, 64, 29)) +>T : Symbol(T, Decl(recursiveMappedTypes.ts, 64, 21)) + +type Child = { [P in NonOptionalKeys]: T[P] } +>Child : Symbol(Child, Decl(recursiveMappedTypes.ts, 64, 90)) +>T : Symbol(T, Decl(recursiveMappedTypes.ts, 65, 11)) +>P : Symbol(P, Decl(recursiveMappedTypes.ts, 65, 19)) +>NonOptionalKeys : Symbol(NonOptionalKeys, Decl(recursiveMappedTypes.ts, 60, 26)) +>T : Symbol(T, Decl(recursiveMappedTypes.ts, 65, 11)) +>T : Symbol(T, Decl(recursiveMappedTypes.ts, 65, 11)) +>P : Symbol(P, Decl(recursiveMappedTypes.ts, 65, 19)) + +export interface ListWidget { +>ListWidget : Symbol(ListWidget, Decl(recursiveMappedTypes.ts, 65, 51)) + + "type": "list", +>"type" : Symbol(ListWidget["type"], Decl(recursiveMappedTypes.ts, 67, 29)) + + "minimum_count": number, +>"minimum_count" : Symbol(ListWidget["minimum_count"], Decl(recursiveMappedTypes.ts, 68, 19)) + + "maximum_count": number, +>"maximum_count" : Symbol(ListWidget["maximum_count"], Decl(recursiveMappedTypes.ts, 69, 28)) + + "collapsable"?: boolean, //default to false, means all expanded +>"collapsable" : Symbol(ListWidget["collapsable"], Decl(recursiveMappedTypes.ts, 70, 28)) + + "each": Child; +>"each" : Symbol(ListWidget["each"], Decl(recursiveMappedTypes.ts, 71, 28)) +>Child : Symbol(Child, Decl(recursiveMappedTypes.ts, 64, 90)) +>ListWidget : Symbol(ListWidget, Decl(recursiveMappedTypes.ts, 65, 51)) +} + +type ListChild = Child +>ListChild : Symbol(ListChild, Decl(recursiveMappedTypes.ts, 73, 1)) +>Child : Symbol(Child, Decl(recursiveMappedTypes.ts, 64, 90)) +>ListWidget : Symbol(ListWidget, Decl(recursiveMappedTypes.ts, 65, 51)) + +declare let x: ListChild; +>x : Symbol(x, Decl(recursiveMappedTypes.ts, 77, 11)) +>ListChild : Symbol(ListChild, Decl(recursiveMappedTypes.ts, 73, 1)) + +x.type; +>x : Symbol(x, Decl(recursiveMappedTypes.ts, 77, 11)) + diff --git a/tests/baselines/reference/recursiveMappedTypes.types b/tests/baselines/reference/recursiveMappedTypes.types index 126d1e1d740..34cfd2d6100 100644 --- a/tests/baselines/reference/recursiveMappedTypes.types +++ b/tests/baselines/reference/recursiveMappedTypes.types @@ -97,3 +97,39 @@ type a = Remap1; // string[] type b = Remap2; // string[] >b : string[] +// Repro from #29992 + +type NonOptionalKeys = { [P in keyof T]: undefined extends T[P] ? never : P }[keyof T]; +>NonOptionalKeys : { [P in keyof T]: undefined extends T[P] ? never : P; }[keyof T] + +type Child = { [P in NonOptionalKeys]: T[P] } +>Child : Child + +export interface ListWidget { + "type": "list", +>"type" : "list" + + "minimum_count": number, +>"minimum_count" : number + + "maximum_count": number, +>"maximum_count" : number + + "collapsable"?: boolean, //default to false, means all expanded +>"collapsable" : boolean + + "each": Child; +>"each" : Child +} + +type ListChild = Child +>ListChild : Child + +declare let x: ListChild; +>x : Child + +x.type; +>x.type : any +>x : Child +>type : any +