TypeScript Version: 3.1
Search Terms: overload all errors
Code
declare function render(s: object, opts: { fileName: string }): void;
declare function render(s: string, name: string): void;
declare const obj: object;
// Actual error: cannot convert 'object' to 'string'
// Desired error: fileName / filename mismatch
render(obj, { filename: "foo" });
When multiple overloads with very different signatures exist, the ordering is effectively arbitrary. This creates really bad error messages when the "last" overload wasn't the one you were trying to call.
While some functions have an extremely large number of overloads, when there are only a few (< 5?) overloads, we should report their failures individually so that hopefully one of the error messages is useful.
e.g. instead of
TS2345: Argument of type 'object' is not assignable to parameter of type 'string'.
it should be
TSXXXX: Failed to find a suitable overload for this call
When invoking "(s: string, name: stri...":
TS2345: Argument of type 'object' is not assignable to parameter of type 'string'.
When invoking "(s: object, opts: { fl...":
TS2345: Argument of type '{ filename: string; }' is not assignable to parameter of type '{ fileName: string; }'.
Object literal may only specify known properties, but 'filename' does not exist in type '{ fileName: string; }'. Did you mean to write 'fileName'?
Playground Link: Link
Related Issues: #26633, others certainly exist
See also https://github.com/garybernhardt/preact-typescript-component-wont-build
+1 on the suggestion.
Also, is there any overlap with @DanielRosenwasser's work on union error messages and overlappy-types? (#27087) Could similar ideas be used to try and report the overload the user probably meant.
In the example, there is an overlap between the first argument for the first overload, and no overlap for the second overload: so the user probably mean the first one.
For small numbers of overloads I think showing all the errors is best as it's quite informative to see TypeScript's reasoning for pruning overloads. My suggestion is really aimed at the larger cases where this wouldn't be tractable.
Basic notes:
Currently not in scope: Reporting for every plausible overload
We are running into this issue with preactjs and fixed it by swapping the overloads around. It seems more like a workaround, but it does fix the issue for us for now. Choosing the declaration with the closest arity seems like the best bet to me. In our case the problematic function has this definition (simplified):
interface ComponentFactory<P> {
props: P;
}
interface Attributes {}
function h<P>(
node: ComponentFactory<P>,
params: Attributes & P | null,
...children: any[]
): any;
function h(
node: string,
params: JSX.HTMLAttributes & JSX.SVGAttributes & Attributes | null,
...children: any[]
): any;
Came across this issue with the following code:
type Options = {
x: number
}
declare function foo(a: string, b: Options): void;
declare function foo(a: number, b: Options): void;
foo('string', { y: 1 });
The error is that the options provided are incorrect, not that string isn't assignable to number. It seems like any functions which take options and have overloads will have poor error reporting when users provide incorrect options.
Thanks for fixing this, @sandersn!
Most helpful comment
Came across this issue with the following code:
The error is that the options provided are incorrect, not that string isn't assignable to number. It seems like any functions which take options and have overloads will have poor error reporting when users provide incorrect options.