Skip to content

Commit c52e598

Browse files
authored
Merge pull request #29068 from Microsoft/noGenericEmptyObject
Generic types should never be considered empty objects
2 parents f2806cd + b454598 commit c52e598

File tree

5 files changed

+123
-2
lines changed

5 files changed

+123
-2
lines changed

src/compiler/checker.ts

Lines changed: 2 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -11386,7 +11386,7 @@ namespace ts {
1138611386
}
1138711387

1138811388
function isEmptyObjectType(type: Type): boolean {
11389-
return type.flags & TypeFlags.Object ? isEmptyResolvedType(resolveStructuredTypeMembers(<ObjectType>type)) :
11389+
return type.flags & TypeFlags.Object ? !isGenericMappedType(type) && isEmptyResolvedType(resolveStructuredTypeMembers(<ObjectType>type)) :
1139011390
type.flags & TypeFlags.NonPrimitive ? true :
1139111391
type.flags & TypeFlags.Union ? some((<UnionType>type).types, isEmptyObjectType) :
1139211392
type.flags & TypeFlags.Intersection ? every((<UnionType>type).types, isEmptyObjectType) :
@@ -12361,7 +12361,7 @@ namespace ts {
1236112361
}
1236212362
else {
1236312363
// An empty object type is related to any mapped type that includes a '?' modifier.
12364-
if (isPartialMappedType(target) && !isGenericMappedType(source) && isEmptyObjectType(source)) {
12364+
if (isPartialMappedType(target) && isEmptyObjectType(source)) {
1236512365
return Ternary.True;
1236612366
}
1236712367
if (isGenericMappedType(target)) {
Lines changed: 41 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,41 @@
1+
//// [genericIsNeverEmptyObject.ts]
2+
// Repro from #29067
3+
4+
function test<T extends { a: string }>(obj: T) {
5+
let { a, ...rest } = obj;
6+
return { ...rest, b: a };
7+
}
8+
9+
let o1 = { a: 'hello', x: 42 };
10+
let o2: { b: string, x: number } = test(o1);
11+
12+
13+
//// [genericIsNeverEmptyObject.js]
14+
"use strict";
15+
// Repro from #29067
16+
var __assign = (this && this.__assign) || function () {
17+
__assign = Object.assign || function(t) {
18+
for (var s, i = 1, n = arguments.length; i < n; i++) {
19+
s = arguments[i];
20+
for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p))
21+
t[p] = s[p];
22+
}
23+
return t;
24+
};
25+
return __assign.apply(this, arguments);
26+
};
27+
var __rest = (this && this.__rest) || function (s, e) {
28+
var t = {};
29+
for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0)
30+
t[p] = s[p];
31+
if (s != null && typeof Object.getOwnPropertySymbols === "function")
32+
for (var i = 0, p = Object.getOwnPropertySymbols(s); i < p.length; i++) if (e.indexOf(p[i]) < 0)
33+
t[p[i]] = s[p[i]];
34+
return t;
35+
};
36+
function test(obj) {
37+
var a = obj.a, rest = __rest(obj, ["a"]);
38+
return __assign({}, rest, { b: a });
39+
}
40+
var o1 = { a: 'hello', x: 42 };
41+
var o2 = test(o1);
Lines changed: 33 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,33 @@
1+
=== tests/cases/compiler/genericIsNeverEmptyObject.ts ===
2+
// Repro from #29067
3+
4+
function test<T extends { a: string }>(obj: T) {
5+
>test : Symbol(test, Decl(genericIsNeverEmptyObject.ts, 0, 0))
6+
>T : Symbol(T, Decl(genericIsNeverEmptyObject.ts, 2, 14))
7+
>a : Symbol(a, Decl(genericIsNeverEmptyObject.ts, 2, 25))
8+
>obj : Symbol(obj, Decl(genericIsNeverEmptyObject.ts, 2, 39))
9+
>T : Symbol(T, Decl(genericIsNeverEmptyObject.ts, 2, 14))
10+
11+
let { a, ...rest } = obj;
12+
>a : Symbol(a, Decl(genericIsNeverEmptyObject.ts, 3, 9))
13+
>rest : Symbol(rest, Decl(genericIsNeverEmptyObject.ts, 3, 12))
14+
>obj : Symbol(obj, Decl(genericIsNeverEmptyObject.ts, 2, 39))
15+
16+
return { ...rest, b: a };
17+
>rest : Symbol(rest, Decl(genericIsNeverEmptyObject.ts, 3, 12))
18+
>b : Symbol(b, Decl(genericIsNeverEmptyObject.ts, 4, 21))
19+
>a : Symbol(a, Decl(genericIsNeverEmptyObject.ts, 3, 9))
20+
}
21+
22+
let o1 = { a: 'hello', x: 42 };
23+
>o1 : Symbol(o1, Decl(genericIsNeverEmptyObject.ts, 7, 3))
24+
>a : Symbol(a, Decl(genericIsNeverEmptyObject.ts, 7, 10))
25+
>x : Symbol(x, Decl(genericIsNeverEmptyObject.ts, 7, 22))
26+
27+
let o2: { b: string, x: number } = test(o1);
28+
>o2 : Symbol(o2, Decl(genericIsNeverEmptyObject.ts, 8, 3))
29+
>b : Symbol(b, Decl(genericIsNeverEmptyObject.ts, 8, 9))
30+
>x : Symbol(x, Decl(genericIsNeverEmptyObject.ts, 8, 20))
31+
>test : Symbol(test, Decl(genericIsNeverEmptyObject.ts, 0, 0))
32+
>o1 : Symbol(o1, Decl(genericIsNeverEmptyObject.ts, 7, 3))
33+
Lines changed: 36 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
=== tests/cases/compiler/genericIsNeverEmptyObject.ts ===
2+
// Repro from #29067
3+
4+
function test<T extends { a: string }>(obj: T) {
5+
>test : <T extends { a: string; }>(obj: T) => Pick<T, Exclude<keyof T, "a">> & { b: string; }
6+
>a : string
7+
>obj : T
8+
9+
let { a, ...rest } = obj;
10+
>a : string
11+
>rest : Pick<T, Exclude<keyof T, "a">>
12+
>obj : T
13+
14+
return { ...rest, b: a };
15+
>{ ...rest, b: a } : Pick<T, Exclude<keyof T, "a">> & { b: string; }
16+
>rest : Pick<T, Exclude<keyof T, "a">>
17+
>b : string
18+
>a : string
19+
}
20+
21+
let o1 = { a: 'hello', x: 42 };
22+
>o1 : { a: string; x: number; }
23+
>{ a: 'hello', x: 42 } : { a: string; x: number; }
24+
>a : string
25+
>'hello' : "hello"
26+
>x : number
27+
>42 : 42
28+
29+
let o2: { b: string, x: number } = test(o1);
30+
>o2 : { b: string; x: number; }
31+
>b : string
32+
>x : number
33+
>test(o1) : Pick<{ a: string; x: number; }, "x"> & { b: string; }
34+
>test : <T extends { a: string; }>(obj: T) => Pick<T, Exclude<keyof T, "a">> & { b: string; }
35+
>o1 : { a: string; x: number; }
36+
Lines changed: 11 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
// @strict: true
2+
3+
// Repro from #29067
4+
5+
function test<T extends { a: string }>(obj: T) {
6+
let { a, ...rest } = obj;
7+
return { ...rest, b: a };
8+
}
9+
10+
let o1 = { a: 'hello', x: 42 };
11+
let o2: { b: string, x: number } = test(o1);

0 commit comments

Comments
 (0)