Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: support error when comparing with object/array literals #45978

Merged
merged 5 commits into from
May 12, 2022

Conversation

Jack-Works
Copy link
Contributor

image

@typescript-bot typescript-bot added the For Uncommitted Bug PR for untriaged, rejected, closed or missing bug label Sep 21, 2021
@typescript-bot
Copy link
Collaborator

This PR doesn't have any linked issues. Please open an issue that references this PR. From there we can discuss and prioritise.

@sandersn
Copy link
Member

@Jack-Works please create an issue to track this. I believe @DanielRosenwasser has some experience about what issues we need to consider for adding a check like this.

@DanielRosenwasser
Copy link
Member

DanielRosenwasser commented Sep 29, 2021

Honestly, if the thing on either side is an explicit literal, I don't think there's ever an issue with providing an error.

@Jack-Works
Copy link
Contributor Author

Do I still need to create an issue since @DanielRosenwasser replied?

Copy link
Member

@sandersn sandersn left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Request for one more test

(And I learned that this applies to ==, not just ===)

src/compiler/checker.ts Outdated Show resolved Hide resolved
@Jack-Works
Copy link
Contributor Author

Request for one more test

(And I learned that this applies to ==, not just ===)

What test do you want to have?

Yes, it applies to == to, although JavaScript can implicit doing a type coherence make the comparison true, TypeScript doesn't consider that case already so I guess it is ok.

if (0 == false) {
    // This condition will always return 'false' since the types 'number' and 'boolean' have no overlap.(2367)
    // but actually true in JavaScript
}

@sandersn
Copy link
Member

sandersn commented Oct 1, 2021

I want to test something that is (1) an existing type error (2) the new object/array literal error:

const b = [1]
if ({ a : 1 } == b) { }

That's the only addition.
My other comments were just about my surprise at learning that == doesn't coerce to string like I thought it did.

@Jack-Works
Copy link
Contributor Author

I have resolved the conflict, can you give another review? thanks, @sandersn @DanielRosenwasser

@sandersn sandersn self-assigned this Feb 19, 2022
@Jack-Works Jack-Works force-pushed the compare-by-literal branch from 29679df to 00b2084 Compare May 11, 2022 15:49
@Jack-Works
Copy link
Contributor Author

Thanks! I have rebased again to fix merge conflicts.

@sandersn
Copy link
Member

Looks like a couple of tests still need their baselines updated.

@sandersn sandersn merged commit b689cd0 into microsoft:main May 12, 2022
@DanielRosenwasser
Copy link
Member

Should we...always show this to JavaScript users?

@Jack-Works
Copy link
Contributor Author

Should we...always show this to JavaScript users?

Of course, should I do anything to enable this in JavaScript?

@Jack-Works Jack-Works deleted the compare-by-literal branch May 13, 2022 02:46
@sandersn
Copy link
Member

sandersn commented May 17, 2022

@Jack-Works there is a set of errors to show to JS users in src/compiler/program.ts .. plainJSErrors, I think. Add it there and it will show up for all JS users. There might be cases where == could succeed that might not be appropriate to show, however, so the check might need to be more complex for JS files.

@Jack-Works
Copy link
Contributor Author

@Jack-Works there is a set of errors to show to JS users in src/compiler/program.ts .. plainJSErrors, I think. Add it there and it will show up for all JS users. There might be cases where == could succeed that might not be appropriate to show, however, so the check might need to be more complex for JS files.

There're only binder errors and grammar errors in plainJSErrors. Is it ok to add type errors?

@sandersn
Copy link
Member

Technically, yes. Grammar errors already come from the checker.
Practically, we need to be really certain the error is correct in order to show it to Javascript users. Syntax and grammar errors are good because they're always runtime errors. I would need to re-read the spec on == to understand whether that's true there.

I vote that we do this for 4.9, after Typescript users have had a chance to report any problems with it. In the past, JS users have gotten irate when code they know to be correct gets a TS error put on it.

@Jack-Works
Copy link
Contributor Author

I can't think any case that == will return true for either side is literal object 🤔 I will open a PR for it

@DanielRosenwasser
Copy link
Member

DanielRosenwasser commented May 18, 2022

It could be surfaced as a suggestion, right? I do agree that we should be a little conservative on things like this for JS users.

@Jack-Works
Copy link
Contributor Author

It could be surfaced as a suggestion, right? I do agree that we should be a little conservative on things like this for JS users.

But that's definitely a mistake even you're using JS 🤔

@sandersn
Copy link
Member

Also, suggestions ought to provide a codefix. And their default UI in vscode is extremely subtle.

@Jack-Works
Copy link
Contributor Author

In this case I cannot provide a code fix because it really depends on the intension of the code 🤔

@sandersn
Copy link
Member

I think we should ship this as an error in 4.9 for JS, perhaps with some more restrictions to cut down on false positives.

@Jack-Works
Copy link
Contributor Author

Can you give an example of false positives on this error? 😂 I can't think any of it

@sandersn
Copy link
Member

I commented on the PR. So far they all look like intentionally bad code, so there might not be anything to do.

@bradzacher
Copy link
Contributor

I can't think any case that == will return true for either side is literal object 🤔

@Jack-Works

[1,2,3] == '1,2,3'
// -> true

({toString() { return 1 }}) == 1
// -> true

== is super sketchy due to its coercion
IIRC the algorithm is something like:

  • if both sides are objects, compare by reference
  • if both sides are the same type, compare exactly
  • else stringify both sides and compare the strings

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
For Uncommitted Bug PR for untriaged, rejected, closed or missing bug
Projects
Archived in project
Development

Successfully merging this pull request may close these issues.

5 participants