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

es private fields in in #52

Merged

Conversation

acutmore
Copy link

@acutmore acutmore commented Feb 23, 2021

*Issue number of the reported bug or feature request: *
microsoft#42574

Describe your changes
Implements the stage 3 'private-fields-in-in' proposal

https://github.com/tc39/proposal-private-fields-in-in

WIP Progress

  • Add new 'PrivateIdentifierInInExpression' expression
    • 'prop' in obj is a BinaryExpression, but this isn't because syntactically the left side can only be a PrivateIdentifier
    • suggestions for alternative names instead of 'PrivateIdentifierInInExpression' ?
  • Error when the PrivateIdentifier can't be looked-up symbolically
    • outside of a class
    • no containing class with matching private field
  • Expression returns boolean
  • RHS must be object or any (and not null in strictNullChecks)
    • i.e. same rules as 'prop' in obj
  • Add new error messages instead of piggybacking on existing diagnostics
  • Use in flowNodes and type narrowing
  • language server hints
    • suggest completion for privateFields on typing #
  • emit ESNext
  • emit down-level
  • part of find references results
  • private static field/method compatibility

@acutmore acutmore force-pushed the es-private-fields-in-in branch from 0cf6e7d to 30f4d03 Compare February 23, 2021 15:20
@acutmore acutmore marked this pull request as draft February 23, 2021 15:21
@acutmore acutmore force-pushed the es-private-fields-in-in branch from e76ef1b to d603860 Compare February 25, 2021 18:35
const id = parsePrivateIdentifier();
if (token() !== SyntaxKind.InKeyword) {
// TODO(aclaymore) use better error
return createMissingNode(SyntaxKind.InKeyword, /*reportAtCurrentPosition*/ true, Diagnostics._0_expected, tokenToString(SyntaxKind.InKeyword));

Choose a reason for hiding this comment

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

Maybe we should think of common parsing mistakes like if(#prop.a in foo). This currently issues the error 'in' expected.. I would argue an error along the lines of: You might have meant this.#props.a would be better,

Copy link
Author

Choose a reason for hiding this comment

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

Looking into this there are existing errors and quickFixes for 'bare' privateFields but my branch was preventing them from running as I was consuming the field in the parser. I've changed the parser to lookahead and only match when the in keyword is present.

Comment on lines 107 to 117
if (#p1 in b) {
b; // good b is 'Bar & Foo'
} else {
b; // good b is Bar
}

Choose a reason for hiding this comment

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

I wonder if we shouldn't be more strict here. I mean for b instanceof Foo it makes sense since you have Symbol.species which can customize how instanceof works. But for #p1 in b there is not universe in which #p1 can be in b and it also be an instance of Bar (if Bar is a class). For other object types this still makes sense... so I'm rather torn if we should issue an error or not.

Copy link
Author

Choose a reason for hiding this comment

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

Interesting. I'm thinking not unless the rule is applied more generally, as in all productions of ClassA & ClassB would be never if at least one class had private fields, and the other was not a sub/superType.

Surprisingly there are cases where the runtime type here could be Bar and not Foo, though I wouldn't expect this to be common code:

class SuperReturn {
  constructor(v) { return v; }
}
class C extends SuperReturn {
  #priv;
  static isC(v) { return #priv in v }
  cMethod() {}
}
class Unrelated {
  random = 1;
  uMethod();
}

const u = new C(new Unrelated));
if (C.isC(u)) {
  u.random; // 1
  u.uMethod(); // fine
  u.cMethod(); // throws
}

@acutmore
Copy link
Author

acutmore commented Mar 9, 2021

Another thing I've noticed is that #priv in v is not coming up in 'find all references' searches. So I need to fix that too.
EDIT: done

acutmore added 20 commits March 29, 2021 11:23
Signed-off-by: Ashley Claymore <[email protected]>
Signed-off-by: Ashley Claymore <[email protected]>
Signed-off-by: Ashley Claymore <[email protected]>
Signed-off-by: Ashley Claymore <[email protected]>
Signed-off-by: Ashley Claymore <[email protected]>
Signed-off-by: Ashley Claymore <[email protected]>
Signed-off-by: Ashley Claymore <[email protected]>
Signed-off-by: Ashley Claymore <[email protected]>
Signed-off-by: Ashley Claymore <[email protected]>
Signed-off-by: Ashley Claymore <[email protected]>
@acutmore acutmore force-pushed the es-private-fields-in-in branch from f0f169e to 42b35c1 Compare March 29, 2021 14:19
Signed-off-by: Ashley Claymore <[email protected]>
Signed-off-by: Ashley Claymore <[email protected]>
Signed-off-by: Ashley Claymore <[email protected]>
Signed-off-by: Ashley Claymore <[email protected]>
Signed-off-by: Ashley Claymore <[email protected]>
@acutmore acutmore force-pushed the es-private-fields-in-in branch from 42b35c1 to 02f59a3 Compare March 30, 2021 11:49
@acutmore acutmore marked this pull request as ready for review March 30, 2021 11:49
@acutmore
Copy link
Author

@dragomirtitian - I've rebased onto TypeScript post the private methods & static fields PR. And ready for another round of review when you have the time.

@acutmore acutmore requested a review from dragomirtitian March 30, 2021 11:56
}
const classSymbol = symbol.parent!;
const classType = <InterfaceType>getTypeOfSymbol(classSymbol);
const firstDecl = symbol.declarations?.[0];

Choose a reason for hiding this comment

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

Wasn't there a valueDeclaration that is what is needed here ?

Copy link
Author

Choose a reason for hiding this comment

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

ah yes there is symbol.valueDeclaration - that looks much better. I'll change to that.

Copy link
Author

Choose a reason for hiding this comment

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

done :)

const firstDecl = symbol.declarations?.[0];
Debug.assert(firstDecl, "should always have a declaration");
let targetType: Type;
if (hasSyntacticModifier(firstDecl, ModifierFlags.Static)) {

Choose a reason for hiding this comment

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

use hasStaticModifier instead

Copy link
Author

Choose a reason for hiding this comment

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

done :)

Comment on lines 23255 to 23257
const ctorSigs = getSignaturesOfType(classType, SignatureKind.Construct);
Debug.assert(ctorSigs.length > 0, "should always have a constructor");
targetType = getReturnTypeOfSignature(ctorSigs[0]);

Choose a reason for hiding this comment

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

There seems to be a better way to get the instance type:

const classSymbol = getSymbolOfNode(classDecl);
const classInstanceType = <InterfaceType>getDeclaredTypeOfSymbol(classSymbol);

At line (24161, fn: classDeclarationExtendsNull)

Not sure if 100% equivalent or if better, but please consider it.

Copy link
Author

Choose a reason for hiding this comment

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

done. Much better, thanks!

@@ -0,0 +1,143 @@
tests/cases/conformance/classes/members/privateNames/privateNameInInExpression.ts(3,27): error TS2805: Static fields with private names can't have initializers when the '--useDefineForClassFields' flag is not specified with a '--target' of 'esnext'. Consider adding the '--useDefineForClassFields' flag.

Choose a reason for hiding this comment

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

Nit: Consider removing this error from the test by changing compiler settings. It is noise here IMO

Copy link
Author

Choose a reason for hiding this comment

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

done :)


const e = #field in (v as never);

for (let f in #field in v as any) { /**/ } // unlikely but valid

Choose a reason for hiding this comment

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

Took a second to get this was actually for (let f in (#field in v as any)) 😅.

Copy link
Author

Choose a reason for hiding this comment

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

haha! I hope no one ever does this but at least we support it if they try

@acutmore acutmore requested a review from mkubilayk May 18, 2021 11:16
Copy link

@mkubilayk mkubilayk left a comment

Choose a reason for hiding this comment

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

Tried it out in VSCode, this is looking good! There is one case I found a bit inconvenient but I'm not sure if we can improve it. When you are inside PrivateIdentifierInInExpression and editing the identifier:

if (# in this) {
     ^ cursor
}

available private identifiers are listed correctly in the suggestions (ctrl+space). But when you pick one if them, you get a property access (this.#foo). If we could use the contextual information to drive this and insert #foo instead, that would be great. If you think people might want this.#foo more often than #foo in this case, please ignore my suggestion.

@@ -3579,6 +3580,9 @@ namespace ts {
return getBinaryOperatorPrecedence(operatorKind);
}

case SyntaxKind.PrivateIdentifierInInExpression:

Choose a reason for hiding this comment

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

nit: This can be combined with SyntaxKind.AsExpression case.

Copy link
Author

Choose a reason for hiding this comment

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

Done :)

@@ -793,6 +793,12 @@ namespace ts {
nodeVisitor(node.operatorToken, tokenVisitor, isBinaryOperatorToken),
nodeVisitor(node.right, visitor, isExpression));

case SyntaxKind.PrivateIdentifierInInExpression:
return factory.updatePrivateIdentifierInInExpression(<PrivateIdentifierInInExpression>node,
nodeVisitor((<PrivateIdentifierInInExpression>node).name, visitor, isExpression),

Choose a reason for hiding this comment

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

Should this use isPrivateIdentifier check?

Suggested change
nodeVisitor((<PrivateIdentifierInInExpression>node).name, visitor, isExpression),
nodeVisitor((<PrivateIdentifierInInExpression>node).name, visitor, isPrivateIdentifier),

Copy link
Author

Choose a reason for hiding this comment

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

If my memory serves correctly: no. As I believe this check is also done on the transformed AST where PrivateIdentifierInInExpression becomes a methodCall.

This was over a month ago so I may be getting this mixed up with a different bit of code. I'll try it out again and confirm either way.

Copy link
Author

Choose a reason for hiding this comment

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

Played around to jog my memory. The reason this was isExpression and not isPrivateIdentifier was because when the identifier is invalid (e.g. used outside of a class) the classFields transformer emits an empty plain Identifier.

I've changed this to isMemberName now to match both private and standard identifiers.

@acutmore
Copy link
Author

Tried it out in VSCode, this is looking good! There is one case I found a bit inconvenient but I'm not sure if we can improve it. When you are inside PrivateIdentifierInInExpression and editing the identifier:

if (# in this) {
     ^ cursor
}

available private identifiers are listed correctly in the suggestions (ctrl+space). But when you pick one if them, you get a property access (this.#foo). If we could use the contextual information to drive this and insert #foo instead, that would be great. If you think people might want this.#foo more often than #foo in this case, please ignore my suggestion.

Yeah I was really unsure on this too. My assumption was that people are more likely to reach for accessing the private field but that's not based on any evidence just a gut feeling.

Maybe it's possible to offer both suggestions a plain #foo identifier when we're preceding an in and the this.#foo. I'll give it a go and see what the API supports.

@acutmore acutmore changed the title es private fields in in [wip] es private fields in in May 24, 2021
- simplify obtaining private fields's static or instance type
- make visitor node test more specific
- clean up warnings about static class fields in tests

Signed-off-by: Ashley Claymore <[email protected]>
@acutmore acutmore force-pushed the es-private-fields-in-in branch from f0bbed9 to 8928039 Compare May 24, 2021 15:03
@acutmore
Copy link
Author

I spent a little bit of time seeing if I could change the completions private fields to sometimes complete to just the identifier but couldn't get anything working quickly. I'll have another go next time I have some time.
I believe I've addressed the other comments.

@acutmore
Copy link
Author

acutmore commented Jun 9, 2021

Merged latest TypeScript main into this branch. All tests still passing.

Copy link

@mkubilayk mkubilayk left a comment

Choose a reason for hiding this comment

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

I think this looks good. Left one comment to prevent against a potential future bug.

We can use microsoft#42458 as an example for the PR description when we open the upstream PR.

* Visits `#id in expr`
*/
function visitPrivateIdentifierInInExpression(node: PrivateIdentifierInInExpression) {
if (!shouldTransformPrivateElements) {

Choose a reason for hiding this comment

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

This is probably fine today but might be a problem when ES2022 is introduced. shouldTransformPrivateElements will probably be updated to say languageVersion < ScriptTarget.ES2022. But it's possible that proposal-private-fields-in-in won't make it to ES2022.

Copy link
Author

Choose a reason for hiding this comment

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

The current plan is to assume https://github.com/tc39/proposal-private-fields-in-in will make stage 4 in time for ES2022. There are positive signals of this, as it is already available in latest Chrome and Firefox.

@acutmore acutmore merged commit f04f22c into bloomberg:private-fields-in-in Jun 17, 2021
molisani pushed a commit to molisani/TypeScript that referenced this pull request Nov 3, 2021
* es private fields in in (bloomberg#52)

add support for the 'private-fields-in-in' TC39 proposal

Signed-off-by: Ashley Claymore <[email protected]>

* [fixup] include inToken when walking forEachChild(node, cb)

* [squash] re-accept lib definition baseline changes

* [squash] reduce if/else to ternary

Signed-off-by: Ashley Claymore <[email protected]>

* [squash] drop 'originalName' and rename parameter instead

Signed-off-by: Ashley Claymore <[email protected]>

* [squash] extend spelling suggestion to all privateIdentifiers

Signed-off-by: Ashley Claymore <[email protected]>

* [squash] revert the added lexical spelling suggestions logic

Signed-off-by: Ashley Claymore <[email protected]>

* [squash] update baseline

Signed-off-by: Ashley Claymore <[email protected]>

* [squash] inline variable as per PR suggestion

Signed-off-by: Ashley Claymore <[email protected]>

* [squash] test targets both esnext and es2020 as per PR comment

Signed-off-by: Ashley Claymore <[email protected]>

* switch to using a binary expression

Signed-off-by: Ashley Claymore <[email protected]>

* [squash] PrivateIdentifier now extends PrimaryExpression

Signed-off-by: Ashley Claymore <[email protected]>

* [squash] accept public api baseline changes

Signed-off-by: Ashley Claymore <[email protected]>

* [squash] classPrivateFieldInHelper now has documentation

Signed-off-by: Ashley Claymore <[email protected]>

* [squash] type-check now follows existing in-expression path

Signed-off-by: Ashley Claymore <[email protected]>

* [squash] parser now follows existing binaryExpression path

Signed-off-by: Ashley Claymore <[email protected]>

* [squash] correct typo in comment

Signed-off-by: Ashley Claymore <[email protected]>

* [squash] no longer use esNext flag

Signed-off-by: Ashley Claymore <[email protected]>

* [squash] swap 'reciever, state' helper params

Signed-off-by: Ashley Claymore <[email protected]>

* [squash] remove change to parenthesizerRules

Signed-off-by: Ashley Claymore <[email protected]>

* [squash] apply suggested changes to checker

Signed-off-by: Ashley Claymore <[email protected]>

* [squash] remove need for assertion in fixSpelling

Signed-off-by: Ashley Claymore <[email protected]>

* [squash] improve comment hint in test

Signed-off-by: Ashley Claymore <[email protected]>

* [squash] fix comment typos

Signed-off-by: Ashley Claymore <[email protected]>

* [squash] add flow-test for Foo | FooSub | Bar

Signed-off-by: Ashley Claymore <[email protected]>

* [squash] add checkExternalEmitHelpers call and new test case

Signed-off-by: Ashley Claymore <[email protected]>

* [squash] simplify and correct parser

Signed-off-by: Ashley Claymore <[email protected]>

* [squash] move most of the added checker logic to expression level

Signed-off-by: Ashley Claymore <[email protected]>

* [squash] always error when privateId could not be resolved

Signed-off-by: Ashley Claymore <[email protected]>

* [squash] reword comment

Signed-off-by: Ashley Claymore <[email protected]>

* [squash] fix codeFixSpelling test

Signed-off-by: Ashley Claymore <[email protected]>

* [squash] do less work

Signed-off-by: Ashley Claymore <[email protected]>

* store symbol by priateId not binaryExpression

Signed-off-by: Ashley Claymore <[email protected]>

* moved parsePrivateIdentifier into parsePrimaryExpression

Signed-off-by: Ashley Claymore <[email protected]>

* [squash] checkInExpressionn bails out early on silentNeverType

Signed-off-by: Ashley Claymore <[email protected]>

* [squash] more detailed error messages

Signed-off-by: Ashley Claymore <[email protected]>

* [squash] resolves conflict in diagnosticMessages.json

Signed-off-by: Ashley Claymore <[email protected]>

* [squash] update baseline for importHelpersES6

Signed-off-by: Ashley Claymore <[email protected]>

* [squash] remove redundent if and comment from parser

Signed-off-by: Ashley Claymore <[email protected]>

* [squash] split up grammar/check/symbolLookup

Signed-off-by: Ashley Claymore <[email protected]>

* [squash] reword message for existing left side of in-expression error

Signed-off-by: Ashley Claymore <[email protected]>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants