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(33367): support destructuring parameter @param tag in tsdoc #50951

Closed
wants to merge 10 commits into from

Conversation

a-tarasyuk
Copy link
Contributor

Fixes #33367

@typescript-bot typescript-bot added the For Backlog Bug PRs that fix a backlog bug label Sep 26, 2022
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.

See previous comments (github PR extension mistakenly posted them early).

@sandersn sandersn self-assigned this Dec 20, 2022
@sandersn
Copy link
Member

Do you want to keep working on this? Otherwise I'd like to close it to reduce the number of open PRs.

@gabritto This is related to the work you're doing on param completions, since it would reduce the complexity of the tags you'd have to emit.

@a-tarasyuk a-tarasyuk requested a review from sandersn March 12, 2023 23:24
@gabritto
Copy link
Member

@typescript-bot pack this

@typescript-bot
Copy link
Collaborator

typescript-bot commented Mar 13, 2023

Heya @gabritto, I've started to run the tarball bundle task on this PR at 4b8a67f. You can monitor the build here.

@typescript-bot
Copy link
Collaborator

typescript-bot commented Mar 13, 2023

Hey @gabritto, I've packed this into an installable tgz. You can install it for testing by referencing it in your package.json like so:

{
    "devDependencies": {
        "typescript": "https://typescript.visualstudio.com/cf7ac146-d525-443c-b23c-0d58337efebc/_apis/build/builds/149208/artifacts?artifactName=tgz&fileId=8B2125470116A7BF9A7289C1BFDDA7736082CFF5910E684C95C227817F09245202&fileName=/typescript-5.1.0-insiders.20230313.tgz"
    }
}

and then running npm install.


There is also a playground for this build and an npm module you can use via "typescript": "npm:@typescript-deploys/[email protected]".;

@a-tarasyuk a-tarasyuk requested review from gabritto and sandersn and removed request for sandersn and gabritto March 13, 2023 20:27
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.

Couple of more requests based on the new tests.

@sandersn
Copy link
Member

Also addresses or even fixes #11859

@gabritto
Copy link
Member

@typescript-bot pack this

@typescript-bot
Copy link
Collaborator

typescript-bot commented Mar 20, 2023

Heya @gabritto, I've started to run the tarball bundle task on this PR at 43cf71b. You can monitor the build here.

@typescript-bot
Copy link
Collaborator

typescript-bot commented Mar 20, 2023

Hey @gabritto, I've packed this into an installable tgz. You can install it for testing by referencing it in your package.json like so:

{
    "devDependencies": {
        "typescript": "https://typescript.visualstudio.com/cf7ac146-d525-443c-b23c-0d58337efebc/_apis/build/builds/150374/artifacts?artifactName=tgz&fileId=E5E88A5F8045188B82960270C715D0A78C138184C23753A31FD491D58E32D32202&fileName=/typescript-5.1.0-insiders.20230320.tgz"
    }
}

and then running npm install.


There is also a playground for this build and an npm module you can use via "typescript": "npm:@typescript-deploys/[email protected]".;

@a-tarasyuk a-tarasyuk requested a review from gabritto March 21, 2023 18:14
Copy link
Member

@gabritto gabritto left a comment

Choose a reason for hiding this comment

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

I unfortunately found more bugs and things we'd like to have before merging the current implementation (JSDoc and destructuring are tricky).

(1) When two or more parameters have the same name, we should probably have a diagnosis (in JS only I think? @sandersn can confirm) on the @param annotation saying the annotation applies to more than one parameter.
Example:

/**
 * @param {string} a - a param // this documents both `a` and the `a` in `{ a: newA }`.
 */
function bar(a, { a: newA }) { }

(2) When checking a function call, we don't get the correct types from the @param annotations following the new logic. Example:

/**
 * @param {number} a - a param
 */
function foo({ a }) {
    a;
}
foo({ a: 1 }); // We incorrectly expect a parameter of type `number` instead of type `{ a: number }`.

The same error also affects signature help info, and in fact the signature of foo anywhere else other than its declaration.

(3) Destructured property has wrong type on hover (might be the same bug as (2), but might be different). Example:

/**
 * @param {string} a - a param
 */
function fizz({ a: newA }) { // `a` has type `any` on hover instead of type `string`

}

Playground with the examples listed above:
https://www.staging-typescript.org/play?ts=5.1.0-pr-50951-7&filetype=js#code/PQKhFgCgAIWgBADgQwE7ILbQN4GcAuqAlgHYDmAvtMtALTXQroZSzBQBmAriQMb5EA9iWgAjNAApkAGhzUAXNBIBTAO4BBaBQCUc4MDGD8AC2gADZGdzAzKjWei9kI48gBuy6Cc+5Mn-ACeiMrSrNDhEZFR0THh+mJc+NCqnirKACbUIsqoqIKo0MJexp5IaJhZJEbIAkXIooKJ0MaCqtBESciIiAA2RMq4XoLQGPn+riLCnkx++Dlh4cgA3AtKauorkBRQUOKoEto7kKAQMHBlzDgkXBiiOVQ09DQzLGfskNx8tSIcgoIS2AYOhwq2WUG2kE4fwBCmgAEYtNoltB4qo8uRGOUsIFgs0csojidWOcXjgCMRyA86AwXsT3p9+EIfkQAF4smHIRR2TTAwHxCwOVyDHGeCwkAIOIotDyoI4Qzis9mAmg6IA

@sandersn
Copy link
Member

The error should be JS-only, but I thought that JSDoc errors only show in JS anyway. Better test that though.

@sandersn
Copy link
Member

sandersn commented Dec 5, 2023

@a-tarasyuk do you want to keep working on this?

@a-tarasyuk
Copy link
Contributor Author

@sandersn Yes., I need some clarification for one case that strikes me as a bit exceptional :).

(2) When checking a function call, we don't get the correct types from the @param annotations following the new logic.

In both cases, we have binding pattern parameters

/**
 * @param {{ a: string, b: string }} foo
 */
function f({ a, b }, { c, d }) { }

/**
 * @param {number} a
 */
function f({ a, b }) { }

How should we determine when TypeScript should strictly use the type from @param and when it should look up @param by the binding element name during the resolution of a parameter type?

TypeScript follows logic wherein, if there is a declared type (in current case jsdoc @param) for a parameter (as described in the lookup logic below), it uses that type. In both scenarios, it resolves the @param tag and uses the type from it.

const declaredType = tryGetTypeFromEffectiveTypeNode(declaration);
...
if (declaredType) {
    return addOptionality(declaredType, isProperty, isOptional);
}

  1. TypeScript jsdoc resolver seeks to find a @param tag by its name
  2. Following that, it attempts to find the @param by its index.

@sandersn
Copy link
Member

As far as I can tell, both calls match tags by index and not by name. The second examples fails because the top-level parameter { a, b } doesn't have a named tag (it doesn't have an identifier for a name anyway), so it falls back to index-based matching, and incorrectly grabs the type number for { a, b }. Is that what you were showing with your example?

I can see two fixes here:

  1. Fancy fix: top-level parameters should skip matching by index if they are a binding pattern and some members of their pattern has a matching tag. This sounds complex and possibly slow.
  2. Dumb fix: binding patterns shouldn't match by index. But this breaks existing code, I think.
  3. Something else: tryGetTypeFromEffectiveTypeNode could pass down a parameter like isBindingPattern that could do the fancy check only when needed, or maybe the fancy check could instead be a recursive attempt to get the effective types of its properties and build an object type for the entire parameter.

For (1) and (2), though, I don't know whether call checking will then work, because @param {number} a only provides a type for the binding pattern property, not the entire binding pattern. This needs to be tested and fixed (somehow?) if it doesn't work.

@a-tarasyuk a-tarasyuk closed this Mar 11, 2025
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
For Backlog Bug PRs that fix a backlog bug
Projects
Status: Done
Development

Successfully merging this pull request may close these issues.

support destructuring parameter @param tag in tsdoc
4 participants