@@ -17770,7 +17770,7 @@ namespace ts {
17770
17770
if (source.flags & TypeFlags.Singleton) return true;
17771
17771
}
17772
17772
if (source.flags & TypeFlags.Object && target.flags & TypeFlags.Object) {
17773
- const related = relation.get(getRelationKey(source, target, IntersectionState.None, relation));
17773
+ const related = relation.get(getRelationKey(source, target, IntersectionState.None, relation, /*ignoreConstraints*/ false ));
17774
17774
if (related !== undefined) {
17775
17775
return !!(related & RelationComparisonResult.Succeeded);
17776
17776
}
@@ -18670,7 +18670,8 @@ namespace ts {
18670
18670
if (overflow) {
18671
18671
return Ternary.False;
18672
18672
}
18673
- const id = getRelationKey(source, target, intersectionState | (inPropertyCheck ? IntersectionState.InPropertyCheck : 0), relation);
18673
+ const keyIntersectionState = intersectionState | (inPropertyCheck ? IntersectionState.InPropertyCheck : 0);
18674
+ const id = getRelationKey(source, target, keyIntersectionState, relation, /*ingnoreConstraints*/ false);
18674
18675
const entry = relation.get(id);
18675
18676
if (entry !== undefined) {
18676
18677
if (reportErrors && entry & RelationComparisonResult.Failed && !(entry & RelationComparisonResult.Reported)) {
@@ -18697,16 +18698,13 @@ namespace ts {
18697
18698
targetStack = [];
18698
18699
}
18699
18700
else {
18700
- // generate a key where all type parameter id positions are replaced with unconstrained type parameter ids
18701
- // this isn't perfect - nested type references passed as type arguments will muck up the indexes and thus
18702
- // prevent finding matches- but it should hit up the common cases
18703
- const broadestEquivalentId = id.split(",").map(i => i.replace(/-\d+/g, (_match, offset: number) => {
18704
- const index = length(id.slice(0, offset).match(/[-=]/g) || undefined);
18705
- return `=${index}`;
18706
- })).join(",");
18701
+ // A key that starts with "*" is an indication that we have type references that reference constrained
18702
+ // type parameters. For such keys we also check against the key we would have gotten if all type parameters
18703
+ // were unconstrained.
18704
+ const broadestEquivalentId = id.startsWith("*") ? getRelationKey(source, target, keyIntersectionState, relation, /*ignoreConstraints*/ true) : undefined;
18707
18705
for (let i = 0; i < maybeCount; i++) {
18708
18706
// If source and target are already being compared, consider them related with assumptions
18709
- if (id === maybeKeys[i] || broadestEquivalentId === maybeKeys[i]) {
18707
+ if (id === maybeKeys[i] || broadestEquivalentId && broadestEquivalentId === maybeKeys[i]) {
18710
18708
return Ternary.Maybe;
18711
18709
}
18712
18710
}
@@ -20261,47 +20259,55 @@ namespace ts {
20261
20259
return isNonDeferredTypeReference(type) && some(getTypeArguments(type), t => !!(t.flags & TypeFlags.TypeParameter) || isTypeReferenceWithGenericArguments(t));
20262
20260
}
20263
20261
20264
- /**
20265
- * getTypeReferenceId(A<T, number, U>) returns "111=0-12=1"
20266
- * where A.id=111 and number.id=12
20267
- */
20268
- function getTypeReferenceId(type: TypeReference, typeParameters: Type[], depth = 0) {
20269
- let result = "" + type.target.id;
20270
- for (const t of getTypeArguments(type)) {
20271
- if (isUnconstrainedTypeParameter(t)) {
20272
- let index = typeParameters.indexOf(t);
20273
- if (index < 0) {
20274
- index = typeParameters.length;
20275
- typeParameters.push(t);
20262
+ function getGenericTypeReferenceRelationKey(source: TypeReference, target: TypeReference, postFix: string, ignoreConstraints: boolean) {
20263
+ const typeParameters: Type[] = [];
20264
+ let constraintMarker = "";
20265
+ const sourceId = getTypeReferenceId(source, 0);
20266
+ const targetId = getTypeReferenceId(target, 0);
20267
+ return `${constraintMarker}${sourceId},${targetId}${postFix}`;
20268
+ // getTypeReferenceId(A<T, number, U>) returns "111=0-12=1"
20269
+ // where A.id=111 and number.id=12
20270
+ function getTypeReferenceId(type: TypeReference, depth = 0) {
20271
+ let result = "" + type.target.id;
20272
+ for (const t of getTypeArguments(type)) {
20273
+ if (t.flags & TypeFlags.TypeParameter) {
20274
+ if (ignoreConstraints || isUnconstrainedTypeParameter(t)) {
20275
+ let index = typeParameters.indexOf(t);
20276
+ if (index < 0) {
20277
+ index = typeParameters.length;
20278
+ typeParameters.push(t);
20279
+ }
20280
+ result += "=" + index;
20281
+ continue;
20282
+ }
20283
+ // We mark type references that reference constrained type parameters such that we know to obtain
20284
+ // and look for a "broadest equivalent key" in the cache.
20285
+ constraintMarker = "*";
20286
+ }
20287
+ else if (depth < 4 && isTypeReferenceWithGenericArguments(t)) {
20288
+ result += "<" + getTypeReferenceId(t as TypeReference, depth + 1) + ">";
20289
+ continue;
20276
20290
}
20277
- result += "=" + index;
20278
- }
20279
- else if (depth < 4 && isTypeReferenceWithGenericArguments(t)) {
20280
- result += "<" + getTypeReferenceId(t as TypeReference, typeParameters, depth + 1) + ">";
20281
- }
20282
- else {
20283
20291
result += "-" + t.id;
20284
20292
}
20293
+ return result;
20285
20294
}
20286
- return result;
20287
20295
}
20288
20296
20289
20297
/**
20290
20298
* To improve caching, the relation key for two generic types uses the target's id plus ids of the type parameters.
20291
20299
* For other cases, the types ids are used.
20292
20300
*/
20293
- function getRelationKey(source: Type, target: Type, intersectionState: IntersectionState, relation: ESMap<string, RelationComparisonResult>) {
20301
+ function getRelationKey(source: Type, target: Type, intersectionState: IntersectionState, relation: ESMap<string, RelationComparisonResult>, ignoreConstraints: boolean ) {
20294
20302
if (relation === identityRelation && source.id > target.id) {
20295
20303
const temp = source;
20296
20304
source = target;
20297
20305
target = temp;
20298
20306
}
20299
20307
const postFix = intersectionState ? ":" + intersectionState : "";
20300
- if (isTypeReferenceWithGenericArguments(source) && isTypeReferenceWithGenericArguments(target)) {
20301
- const typeParameters: Type[] = [];
20302
- return getTypeReferenceId(source as TypeReference, typeParameters) + "," + getTypeReferenceId(target as TypeReference, typeParameters) + postFix;
20303
- }
20304
- return source.id + "," + target.id + postFix;
20308
+ return isTypeReferenceWithGenericArguments(source) && isTypeReferenceWithGenericArguments(target) ?
20309
+ getGenericTypeReferenceRelationKey(source as TypeReference, target as TypeReference, postFix, ignoreConstraints) :
20310
+ `${source.id},${target.id}${postFix}`;
20305
20311
}
20306
20312
20307
20313
// Invoke the callback for each underlying property symbol of the given symbol and return the first
0 commit comments