Typescript: Report errors on all plausible overloads

Created on 21 Sep 2018  路  5Comments  路  Source: microsoft/TypeScript

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

Committed Suggestion

Most helpful comment

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.

All 5 comments

+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:

  • We should first filter by "close" arity
  • Then also filter by some rough estimate of type closeness of all arguments
  • Also include a related span for the overload we selected for the purposes of error reporting

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!

Was this page helpful?
0 / 5 - 0 ratings

Related issues

Taytay picture Taytay  路  174Comments

yortus picture yortus  路  157Comments

rbuckton picture rbuckton  路  139Comments

blakeembrey picture blakeembrey  路  171Comments

rwyborn picture rwyborn  路  210Comments