Skip to content

Performance issue checking IndexedAccess rest parameter that resolves to large union of tuples #26756

Closed

Description

TypeScript Version: 3.1.0-dev.20180829, master

Search Terms: rest, parameters, union, tuple, IndexedAccess,

I realize this may be firmly in the camp of "doctor, it hurts when I do this", but thought I'd still file :)

I ran into this when using a union of tuples in place of overloading, not realizing it wasn't really supposed to be working until #26676. Since the tuples came from the values of an interface, I believe the indirect indexed access was allowing it to still pass since it would fall back to structural checking against any[].

With #26676 landed it's still an issue, though.

Code
This was from an attempt to type the Chrome debugger rpc protocol using tuples to represent that some commands need no parameters, some need one, and some can optionally have one. I'll link to the type files since they're large, but I can provide them elsewhere if needed: ChromeDevTools/devtools-protocol/pull/113/protocol-mapping.d.ts, and the types referenced in that file are found in protocol.d.ts. All the types end up relatively shallow and without cycles.

Poor performance can be triggered with

import ProtocolMapping  from './protocol-mapping';
type Commands = ProtocolMapping.Commands;

function sendCommand<C extends keyof Commands>(method: C, ...params: Commands[C]['paramsType']) {}

This takes ~6 seconds to check with tsc --strict on my machine, with about 3.5 seconds attributable to checkParameter() of ...params. More crucially it adds a similar amount of time to tsserver responses, and the work doesn't seem to be cached, so it re-runs on every intellisense completion.

On the other hand, the not quite equivalent

function sendCommand<C extends keyof Commands>(method: C, ...params: Commands[keyof Commands]['paramsType']) {}

takes about 2 seconds (with about 300 ms of that attributable to checkParameter of ...params).

The difference appears to come down to when the checker realizes the type is a union of tuples. The fast version does appear to end up checking each tuple for structural similarity to any[], but it does so for each tuple in the union individually, which is relatively fast.

The slow version seems to instead get deep enough that it ends up taking the union of each property across the tuples, then compares each of those against the properties of any[]. Since in this example basically each tuple ends up with a unique typeArgument, I assume the structural comparison against these unions for each property is what's taking the bulk of this time. The profile is mostly stuck in getPropertiesOfUnionOrIntersectionType down to createUnionOrIntersectionProperty and then splitting up into many nested isRelatedTo calls.

Hopefully the slow case could be made to function more like the fast case, but it also seems like a shortcut is possible since the tuples are never used for anything but rest parameters (so memoized type info is never used again) and as soon as getApparentType() is called with Commands[C]['paramsType'], the type then becomes a union of objects with ObjectFlags.Tuple, which could be used for an even quicker test.

To check, I added a super hacky isUnionOfTuples method and the time spent dropped to essentially nothing. I might be missing the corner cases here (or that my case is a special case of some tricky general issue), but if rest parameters for overloading becomes popular, it seems like a fast path for when the apparent type of source is a union of tuples and target === anyArrayType could be generally useful without much cost.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Metadata

Assignees

Labels

BugA bug in TypeScriptFixedA PR has been merged for this issue

Type

No type

Projects

No projects

Relationships

None yet

Development

No branches or pull requests

Issue actions