-
Notifications
You must be signed in to change notification settings - Fork 12.8k
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
Import statement completions #43149
Import statement completions #43149
Conversation
Thanks for the PR! It looks like you've changed the TSServer protocol in some way. Please ensure that any changes here don't break consumers of the current TSServer API. For some extra review, we'll ping @sheetalkamat, @amcasey, @mjbvz, @minestarks for you. Feel free to loop in other consumers/maintainers if necessary |
@@ -2127,7 +2127,7 @@ namespace ts.server.protocol { | |||
arguments: FormatOnKeyRequestArgs; | |||
} | |||
|
|||
export type CompletionsTriggerCharacter = "." | '"' | "'" | "`" | "/" | "@" | "<" | "#"; | |||
export type CompletionsTriggerCharacter = "." | '"' | "'" | "`" | "/" | "@" | "<" | "#" | " "; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The space, in combination with the isIncomplete
property added, is a targeted fix for the problem described at #31658 (comment).
function key(name: string, alias: Symbol, moduleSymbol: Symbol, checker: TypeChecker) { | ||
const moduleName = stripQuotes(moduleSymbol.name); | ||
const moduleKey = isExternalModuleNameRelative(moduleName) ? "/" : moduleName; | ||
return `${name}|${getSymbolId(skipAlias(alias, checker))}|${moduleKey}`; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
At the moment, the contents of the map are not meant to be looked up by keying into it; rather, the key serves to manage grouping. Each key maps to a group of exports/re-exports of the same symbol by the same name from a set of modules that are effectively interchangeable until we need to calculate what the best module specifier to use is. In other words, if we’re generating a completions list of auto imports, we should get exactly one completion entry per key, even if that key maps to many re-exports.
/** Information about how a symbol is exported from a module. (We don't need to store the exported symbol, just its module.) */ | ||
interface SymbolExportInfo { | ||
readonly moduleSymbol: Symbol; | ||
readonly importKind: ImportKind; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Some of the refactors here weren’t strictly necessary, but are a step forward in understandability and hopefully cacheability in the future—SymbolExportInfo
previously had info about how a specific file would want to import the exported symbol instead of just plain info about how the symbol is exported, which made it a misnomer and uncacheable.
@@ -82,11 +100,12 @@ namespace ts.Completions { | |||
} | |||
|
|||
/** | |||
* Map from symbol id -> SymbolOriginInfo. | |||
* Map from symbol index in `symbols` -> SymbolOriginInfo. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
A semi-related bug that became apparent in the process here was that the same symbol can have multiple origins (e.g., one symbol exported from two different ambient module declarations is allowed to show up as two separate auto import completions), and keying by symbol id didn’t allow that.
@@ -598,8 +619,55 @@ namespace ts.Completions { | |||
has: name => uniques.has(name), | |||
add: name => uniques.set(name, true), | |||
}; | |||
|
|||
function shouldIncludeSymbol(symbol: Symbol, symbolToSortTextMap: SymbolSortTextMap): boolean { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This was just moved from a separate filter statement elsewhere into this spot where we already iterate through the whole list. Should be a small performance improvement, but I moved it because of some order-of-operations change that I honestly don’t remember anymore.
@@ -1153,7 +1249,7 @@ namespace ts.Completions { | |||
Debug.assertEachIsDefined(tagSymbols, "getJsxIntrinsicTagNames() should all be defined"); | |||
tryGetGlobalSymbols(); | |||
symbols = tagSymbols.concat(symbols); | |||
completionKind = CompletionKind.MemberLike; | |||
completionKind = CompletionKind.Global; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
AFAICT this completionKind
had no effect prior to me moving the aforementioned filtering code, but was conceptually wrong, and changing it to be conceptually right allowed me to move the filter.
@@ -1349,13 +1446,24 @@ namespace ts.Completions { | |||
const nameSymbol = leftMostName && typeChecker.getSymbolAtLocation(leftMostName); | |||
// If this is nested like for `namespace N { export const sym = Symbol(); }`, we'll add the completion for `N`. | |||
const firstAccessibleSymbol = nameSymbol && getFirstSymbolInChain(nameSymbol, contextToken, typeChecker); | |||
if (firstAccessibleSymbol && !symbolToOriginInfoMap[getSymbolId(firstAccessibleSymbol)]) { | |||
if (firstAccessibleSymbol && addToSeen(seenPropertySymbols, getSymbolId(firstAccessibleSymbol))) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Small tweaks here due to changing the origin info map to be keyed by index instead of by symbol id—we now need a Map for deduplication here, since symbol duplication is explicitly allowed in other cases.
Co-authored-by: Daniel Rosenwasser <[email protected]>
Co-authored-by: Daniel Rosenwasser <[email protected]>
Hey @andrewbranch this doesn't appear to be working for me am I doing something wrong? |
It looks like something might have broken on the VS Code side. It doesn't work for me either at the moment, but I can see TS Server returning the right things in the log. I'll take a look. Thanks! |
Have it been considered to add support for that in the LSP spec? Or is it already supported? That question maybe belongs better in microsoft/vscode#119009 but that one is locked. |
I made sure that everything we’re doing with VS Code can also be done over LSP so that this will work in VS once their TS completions support swaps over to LSP. I actually have a draft PR up for that in the TypeScript-VS codebase (which is not open source). I’m not sure if that answers your question—let me know if you have more specific questions, and I can try to answer or pull in @uniqueiniquity who is our resident LSP expert. |
Hey @andrewbranch was that issue ever fixed? Or is that what you need to change in the VSCode repo? |
@jasonwilliams it was fixed at microsoft/vscode#121975 |
I guess the more specific question I have is whether that functionality requires changes to the LSP spec (https://github.com/microsoft/language-server-protocol) and if so whether those were proposed or even already implemented? |
No, this feature can be implemented with LSP 3.16 (and likely earlier versions) without changes. From the protocol standpoint, this is actually a good bit simpler than normal auto imports (though those are possible with LSP too). The only somewhat interesting thing about this feature is the tab stop at the end, which is supported via snippet text. The only caveat is that I made the space character a trigger character only when preceded by |
Screen capture demo
Enables auto-import-style completions on import statements themselves, e.g.
can offer a completion to
(The
|
indicates the position of the cursor both before and after the completion.)This PR is large because, although the feature appears very similar to existing auto imports, it requires us to resolve module specifiers immediately instead of in a subsequent completion details request. Being able to do that with any reasonable amount of code reuse required quite a bit of untangling. The implementation is explained in some detail in code review comments, but note that a large amount of the PR diff is just functions being shuffled around.
Note also this feature is opt-in via a user preference (
includeCompletionsForImportStatements
) because it requires corresponding editor changes. So, you won’t be able to test this yourself without building my VS Code PR as well.Design
esModuleInterop
introduces ambiguity we try to give you what you want)const foo = require("mod")
; only works on imports (thoughimport foo = require("mod")
is supported in TS)import foo, { bar|
orimport { foo, bar|
. This could be relaxed a bit later, but it was complicated and seemed fairly low value.replacementRange
must be a single line), does not work on imports that already span multiple lines, e.g.Performance
Following measurements assume these dependencies
Import statement completions
Stress-tested with
aws-sdk
installed, which has just a ridiculous number of exports. First draft took several seconds without caches, which was too much, so I decided to require the first character of the identifier to match the first character of the export name before continuing the fuzzy match. So whereas regular auto imports will offeruseState
when you have just typedstate
, import statement completions require you to start withu
.import a|
: 865 ms (3604 entries)import b|
: 456 ms(1279 entries) (some cache benefit from previous request)import a|
: 625 ms (max cache benefit)After
npm rm aws-sdk
:import a|
: 116 ms (197 entries)import b|
: 44 ms (105 entries)import a|
: 70 msNormal auto imports
The first draft of this PR incurred about a 15% performance penalty on all auto-imports; that has now been reduced to zero or has improved performance in some scenarios. The auto import cache has been split into two pieces that are independently more durable and more reusable than they were combined.
Previously, the process went something like this:
"lodash"
), do the package.json files visible to the importing file list that package?The main problem with this process is that steps (2) and (3) are pretty expensive, and that work might go to waste after the filter in step (6) is applied. The other problem is that they are dependent on the location of the importing file and the contents of any package.jsons visible to that file, which makes us have to invalidate the cache a lot. Too many inputs combined into a single cache means that when we invalidate the cache, a lot of work that isn’t actually invalid has to be thrown away. Now, we cache “what are the exports and re-exports of every module” and “is file A importable from file B, and by what module specifier” separately. So the process becomes more like:
This means for subsequent auto imports, we have better chances of getting cache hits, and we do less expensive work up front, even with the caches are totally empty.
Triggering auto import completions in a project with
completionInfo
(cold)completionEntryDetails
completionInfo
(cached)completionEntryDetails
To-do
aws-sdk
installed) and probably implement some limits/mitigations there. Increased strictness of filter to make list smaller.Fixes #31658