-
-
Notifications
You must be signed in to change notification settings - Fork 4.7k
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
Proposed rule change: no-unused-vars should consider a var "used" in JSDoc #9050
Comments
Trackback to #7200 which proposes the same change but I guess didn't have enough information to kick off the process? |
Thanks for the proposal. This seems like it might be a good idea (at least as an option), although it might be tricky to handle the edge cases correctly, because JSDoc comments don't have a well-defined variable scope as far as I know. import Foo from 'foo';
{
const Foo = somethingElse;
/**
* @param {Foo} what does this mean?
*/
function x() {}
} |
Would it be appropriate to survey how this is implemented in various JSDoc-consuming toolsets and see if there's a consensus? Just as a data point, VSCode treats the JSDoc param references as being scoped where they're declared:
The |
As a maintainer of closure-compiler, I can pipe in here. The scope of a JSDoc comment is exactly where it occurs. From the above example: import Foo from 'foo';
{
const Foo = somethingElse;
/**
* @param {Foo} what does this mean? -- it means somethingElse. No ambiguity at all.
*/
function x() {}
} |
Hello, what's the status of this? It would be great to have it. |
As much as I would like to see this warning go away, importing a module that isn't used is more of a dirty fix for VSCode. It also has it's own issues. For example, performing an It could create an out of order module loading, therefore the eslint warning/error is a good one. PS: You can use See: microsoft/TypeScript#14377 for a more permanent fix. |
JSDoc is comments; variables aren't "used" by comments, only by code. If this support is added (I don't think it should be; it belongs in a separate JSDoc-related rule, just like jsx vars being marked as used are in an eslint-plugin-react rule), I hope it's at least in a separate option. |
The above comments make good points. Maybe it would make more sense to have a directive specific to JSDoc that allows it to resolve imports that are used in docs, without affecting the actual code? Would something like |
No - the code is not unused. The import is actually needed to order dependencies correctly. And additionally - it's not global. During module references, closure-compiler will update the type nodes as well and the import is required to do that.
This is a case where an alternative scheme should be used. If you just want to mention |
@ChadKillingsworth Documentation isn't used by the compiler. When you load a module, you're actively loading all its dependencies and whatever initialization it wants into runtime. This means more objects into memory and whatever else is going on with the module import process. Using a runtime command for the sake of a transpile process (closure-compiler) isn't appropriate. That is, unless you want the transpile process to remove the import line which doesn't sound safe at all. That's why it should be in comments. Microsoft's Typescript handles the same issue by using triple slashes ( That's why the argument about handling this issue within Microsoft's Salsa (JS Typechecking) leans to using a comment based import. microsoft/TypeScript#14377 But back to the original reported issue at hand, the question is why is @thw0rted even importing a module without referencing it in code in the first place? If he's doing it only to workaround an issue with Microsoft's Salsa typechecking (which I'm guessing he is), he should follow the issue I posted before. If he's using it for Closure transpiling, he should file the issue there. |
This is simply not true for Closure Compiler. It's used heavily.
He is referencing the code in type annotations. In cases where you want to reference a module type without actually importing it, Closure-Compiler supports the syntax: /** @type {./path/to/module.propName} */ However, use of that syntax inappropriately can cause files to be ordered incorrectly when compiling. |
If a compiler uses documentation, that’s fine, but then the compiler is obligated to actually use documentation. If something declared only in comments needs additional information for the compiler to be able to work with it, that info needs to also be in a comment. If Closure Compiler can’t do this, it seems like a bug in that project; not something eslint needs to handle. |
Just to settle the debate, back when I filed this issue I was using eslint, but as a VS Code plugin, so I cared about both eslint and Salsa -- ideally, any solution would work with both of them. I've since moved that codebase to Typescript but I think I still have one or two places where this "kind" of problem comes up. I think from eslint's perspective, specifically, you can always resolve the issue with a simple |
@ChadKillingsworth My reference to compiler was the javascript runtime compiler (ie: V8). You're talking about use with closure-compiler which is a transpiler than that converts code down to the actual JS runtime compiler. The documentation is ignored there. Still, as you said, Closure Compiler has syntax for referencing types via comments. So in reality, you shouldn't be using |
Hi all, this just came up again -- I'm upgrading to Webpack 4.x and discovering that type-checking in the editor would be super helpful. Previously, I just declared
then ran the build tool and hoped the config was valid. There's a typings file for Webpack, of course, which means that there's an interface definition that lays out exactly what's valid in that assignment... if I can explicitly set its type. But to do that, I need to import the config-object interface from the module, and of course that makes eslint unhappy. This isn't a deal-breaker, since I don't have e.g. hooks to reject commits that cause eslint errors, but I'd still really appreciate a way to either a) import something only for docs, or b) mark an import as "actually used even though eslint says it's not", on a one-off basis. |
You can accomplish (b) using inline disable comments. Example: Place |
Thanks, I actually found this while poking around solving other issues -- for some reason I can't get |
Agreed, just wanted to say option b was possible now in case you didn't know. (Not sure why the line disable comments didn't work, though...) |
@platinumazure, I ran into same issue with type declarations. Option (b) simply disables unused imports check at all. And if we remove type declaration usage from the file, we want to see warning about unused import. But ESLint tells us that everything is Ok. |
VSCode (and Typescript) now supports
So you don't have to perform unused imports anymore. |
Thanks, @clshortfuse , I'm tracking via #14377 and #22592. I don't know why I didn't see the links to the former in earlier comments, but it makes sense. |
A thought occurs to me, though: if this syntax is unique to |
@thw0rted Yes. I work with an eslint rule that requires valid jsdoc and this new syntax works fine. |
@ChadKillingsworth do you have any sense of whether Closure might consider adding a notion like the VSCode/Typescript workaround mentioned above ( |
Are you not able to do a single-line eslint-disable comment? That's what I've been using though now that I'm thinking of it, I'll probably go ahead and switch to |
The single-line eslint disable comment doesn't work so well if you have multiple imports on one line: import { Foo, Bar, Baz } from './example.js';
export class Test {
/**
* @param {!Foo} foo
*/
constructor(foo) {
/** @const */
this._foo = foo;
/** @const */
this._bar = new Bar();
}
} Adding the single-line eslint disable comment to line 1 causes the unused import {
Bar,
Baz,
Foo, // eslint-disable-line
} from './one.js'; but that is going to require a large scale refactor and a code convention change. Not impossible, but unideal. |
@timdawborn can't you use import { Bar, Baz } from './example.js';
/** @typedef {import('./example.js').Foo} Foo */
export class Test {
/**
* @param {!Foo} foo
*/
constructor(foo) {
/** @const */
this._foo = foo;
/** @const */
this._bar = new Bar();
}
} |
@LinusU
|
aha, good to know 👍 |
Perhaps an issue should be filed on closure compiler to support that syntax - that seems like it’d both solve this issue and also be consistent with typescript’s usage of jsdoc. |
google/closure-compiler#3041 discusses some of this. |
Workaround for this issue: import Foo from "foo"; // eslint-disable-line no-unused-vars
/**
* @param {Foo} f
*/
function bar(f){ return f.baz; } |
Thanks for the suggestion. As of yesterday, we have decided to officially end-of-life JSDoc support in ESLint. All JSDoc features are now deprecated and we won't be fixing bugs or making any further improvements to those features. We are recommending that people transition over to use the Thanks for understanding and we apologize for any inconvenience. |
What rule do you want to change?
no-unused-vars
Does this change cause the rule to produce more or fewer warnings?
How will the change be implemented? (New option, new default behavior, etc.)?
Please provide some example code that this change will affect:
What does the rule currently do for this code?
Foo
(inimport Foo
) as an unused var.What will the rule do after it's changed?
Foo
as unused because it is referenced in the JSDoc.The text was updated successfully, but these errors were encountered: