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

[Experiment] Some tweaking to handle project references for auto import #55955

Merged
merged 13 commits into from
Apr 18, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
6 changes: 6 additions & 0 deletions src/compiler/utilities.ts
Original file line number Diff line number Diff line change
Expand Up @@ -9064,6 +9064,7 @@ export interface SymlinkCache {
getSymlinkedFiles(): ReadonlyMap<Path, string> | undefined;
setSymlinkedDirectory(symlink: string, real: SymlinkedDirectory | false): void;
setSymlinkedFile(symlinkPath: Path, real: string): void;
hasAnySymlinks(): boolean;
/**
* @internal
* Uses resolvedTypeReferenceDirectives from program instead of from files, since files
Expand Down Expand Up @@ -9118,8 +9119,13 @@ export function createSymlinkCache(cwd: string, getCanonicalFileName: GetCanonic
typeReferenceDirectives.forEach(resolution => processResolution(this, resolution.resolvedTypeReferenceDirective));
},
hasProcessedResolutions: () => hasProcessedResolutions,
hasAnySymlinks,
};

function hasAnySymlinks() {
return !!symlinkedFiles?.size || (!!symlinkedDirectories && !!forEachEntry(symlinkedDirectories, value => !!value));
}

function processResolution(cache: SymlinkCache, resolution: ResolvedModuleFull | ResolvedTypeReferenceDirective | undefined) {
if (!resolution || !resolution.originalPath || !resolution.resolvedFileName) return;
const { resolvedFileName, originalPath } = resolution;
Expand Down
2 changes: 1 addition & 1 deletion src/harness/tsserverLogger.ts
Original file line number Diff line number Diff line change
Expand Up @@ -123,7 +123,7 @@ export function sanitizeLog(s: string): string {
s = s.replace(/getExportInfoMap: done in \d+(?:\.\d+)?/g, `getExportInfoMap: done in *`);
s = s.replace(/collectAutoImports: \d+(?:\.\d+)?/g, `collectAutoImports: *`);
s = s.replace(/continuePreviousIncompleteResponse: \d+(?:\.\d+)?/g, `continuePreviousIncompleteResponse: *`);
s = s.replace(/dependencies in \d+(?:\.\d+)?/g, `dependencies in *`);
s = s.replace(/referenced projects in \d+(?:\.\d+)?/g, `referenced projects in *`);
s = s.replace(/"exportMapKey":\s*"\d+ \d+ /g, match => match.replace(/ \d+ /, ` * `));
s = s.replace(/getIndentationAtPosition: getCurrentSourceFile: \d+(?:\.\d+)?/, `getIndentationAtPosition: getCurrentSourceFile: *`);
s = s.replace(/getIndentationAtPosition: computeIndentation\s*: \d+(?:\.\d+)?/, `getIndentationAtPosition: computeIndentation: *`);
Expand Down
26 changes: 24 additions & 2 deletions src/server/editorServices.ts
Original file line number Diff line number Diff line change
Expand Up @@ -1700,6 +1700,16 @@ export class ProjectService {
const project = this.getConfiguredProjectByCanonicalConfigFilePath(projectCanonicalPath);
if (!project) return;

if (
configuredProjectForConfig !== project &&
this.getHostPreferences().includeCompletionsForModuleExports
) {
const path = this.toPath(configFileName);
if (find(project.getCurrentProgram()?.getResolvedProjectReferences(), ref => ref?.sourceFile.path === path)) {
project.markAutoImportProviderAsDirty();
}
}

// Load root file names for configured project with the config file name
// But only schedule update if project references this config file
const updateLevel = configuredProjectForConfig === project ? ProgramUpdateLevel.RootNamesAndUpdate : ProgramUpdateLevel.Update;
Expand Down Expand Up @@ -1765,11 +1775,19 @@ export class ProjectService {
project.pendingUpdateLevel = ProgramUpdateLevel.Full;
project.pendingUpdateReason = loadReason;
this.delayUpdateProjectGraph(project);
project.markAutoImportProviderAsDirty();
}
else {
// Change in referenced project config file
project.resolutionCache.removeResolutionsFromProjectReferenceRedirects(this.toPath(canonicalConfigFilePath));
const path = this.toPath(canonicalConfigFilePath);
project.resolutionCache.removeResolutionsFromProjectReferenceRedirects(path);
this.delayUpdateProjectGraph(project);
if (
this.getHostPreferences().includeCompletionsForModuleExports &&
find(project.getCurrentProgram()?.getResolvedProjectReferences(), ref => ref?.sourceFile.path === path)
) {
project.markAutoImportProviderAsDirty();
}
}
});
return scheduledAnyProjectUpdate;
Expand Down Expand Up @@ -3557,6 +3575,7 @@ export class ProjectService {
const {
lazyConfiguredProjectsFromExternalProject,
includePackageJsonAutoImports,
includeCompletionsForModuleExports,
} = this.hostConfiguration.preferences;

this.hostConfiguration.preferences = { ...this.hostConfiguration.preferences, ...args.preferences };
Expand All @@ -3576,7 +3595,10 @@ export class ProjectService {
})
);
}
if (includePackageJsonAutoImports !== args.preferences.includePackageJsonAutoImports) {
if (
includePackageJsonAutoImports !== args.preferences.includePackageJsonAutoImports ||
!!includeCompletionsForModuleExports !== !!args.preferences.includeCompletionsForModuleExports
) {
this.forEachProject(project => {
project.onAutoImportProviderSettingsChanged();
});
Expand Down
88 changes: 69 additions & 19 deletions src/server/project.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ import {
arrayToMap,
BuilderState,
CachedDirectoryStructureHost,
changeExtension,
changesAffectModuleResolution,
clearMap,
cloneCompilerOptions,
Expand All @@ -16,7 +17,6 @@ import {
comparePaths,
CompilerHost,
CompilerOptions,
concatenate,
containsPath,
createCacheableExportInfoMap,
createLanguageService,
Expand Down Expand Up @@ -50,6 +50,7 @@ import {
getAutomaticTypeDirectiveNames,
getBaseFileName,
GetCanonicalFileName,
getCommonSourceDirectoryOfConfig,
getDeclarationEmitOutputFilePathWorker,
getDefaultCompilerOptions,
getDefaultLibFileName,
Expand All @@ -60,6 +61,7 @@ import {
getEntrypointsFromPackageJsonInfo,
getNormalizedAbsolutePath,
getOrUpdate,
getOutputDeclarationFileName,
GetPackageJsonEntrypointsHost,
getStringComparer,
HasInvalidatedLibResolutions,
Expand All @@ -79,6 +81,7 @@ import {
map,
mapDefined,
maybeBind,
memoize,
ModuleResolutionCache,
ModuleResolutionHost,
noop,
Expand Down Expand Up @@ -1316,6 +1319,12 @@ export abstract class Project implements LanguageServiceHost, ModuleResolutionHo
}
}

/** @internal */
markAutoImportProviderAsDirty() {
if (!this.autoImportProviderHost) this.autoImportProviderHost = undefined;
this.autoImportProviderHost?.markAsDirty();
}

/** @internal */
onAutoImportProviderSettingsChanged() {
if (this.autoImportProviderHost === false) {
Expand Down Expand Up @@ -1400,8 +1409,7 @@ export abstract class Project implements LanguageServiceHost, ModuleResolutionHo
this.projectProgramVersion++;
}
if (hasAddedorRemovedFiles) {
if (!this.autoImportProviderHost) this.autoImportProviderHost = undefined;
this.autoImportProviderHost?.markAsDirty();
this.markAutoImportProviderAsDirty();
}
if (isFirstProgramLoad) {
// Preload auto import provider so it's not created during completions request
Expand Down Expand Up @@ -2439,7 +2447,7 @@ export class AutoImportProviderProject extends Project {

const start = timestamp();
let dependencyNames: Set<string> | undefined;
let rootNames: string[] | undefined;
let rootNames: Set<string> | undefined;
const rootFileName = combinePaths(hostProject.currentDirectory, inferredTypesContainingFile);
const packageJsons = hostProject.getPackageJsonsForAutoImport(combinePaths(hostProject.currentDirectory, rootFileName));
for (const packageJson of packageJsons) {
Expand Down Expand Up @@ -2471,8 +2479,7 @@ export class AutoImportProviderProject extends Project {
if (packageJson) {
const entrypoints = getRootNamesFromPackageJson(packageJson, program, symlinkCache);
if (entrypoints) {
rootNames = concatenate(rootNames, entrypoints);
dependenciesAdded += entrypoints.length ? 1 : 0;
dependenciesAdded += addRootNames(entrypoints);
continue;
}
}
Expand All @@ -2490,8 +2497,7 @@ export class AutoImportProviderProject extends Project {
);
if (typesPackageJson) {
const entrypoints = getRootNamesFromPackageJson(typesPackageJson, program, symlinkCache);
rootNames = concatenate(rootNames, entrypoints);
dependenciesAdded += entrypoints?.length ? 1 : 0;
dependenciesAdded += addRootNames(entrypoints);
return true;
}
}
Expand All @@ -2504,16 +2510,56 @@ export class AutoImportProviderProject extends Project {
// package and load the JS.
if (packageJson && compilerOptions.allowJs && compilerOptions.maxNodeModuleJsDepth) {
const entrypoints = getRootNamesFromPackageJson(packageJson, program, symlinkCache, /*resolveJs*/ true);
rootNames = concatenate(rootNames, entrypoints);
dependenciesAdded += entrypoints?.length ? 1 : 0;
dependenciesAdded += addRootNames(entrypoints);
}
}
}

if (rootNames?.length) {
hostProject.log(`AutoImportProviderProject: found ${rootNames.length} root files in ${dependenciesAdded} dependencies in ${timestamp() - start} ms`);
const references = program.getResolvedProjectReferences();
let referencesAddded = 0;
if (references?.length && hostProject.projectService.getHostPreferences().includeCompletionsForModuleExports) {
// Add direct referenced projects to rootFiles names
references.forEach(ref => {
if (ref?.commandLine.options.outFile) {
referencesAddded += addRootNames(filterEntrypoints([
changeExtension(ref.commandLine.options.outFile, ".d.ts"),
]));
}
else if (ref) {
const getCommonSourceDirectory = memoize(() =>
getCommonSourceDirectoryOfConfig(
ref.commandLine,
!hostProject.useCaseSensitiveFileNames(),
)
);
referencesAddded += addRootNames(filterEntrypoints(mapDefined(
ref.commandLine.fileNames,
fileName =>
!isDeclarationFileName(fileName) &&
!fileExtensionIs(fileName, Extension.Json) &&
!program.getSourceFile(fileName) ?
getOutputDeclarationFileName(
fileName,
ref.commandLine,
!hostProject.useCaseSensitiveFileNames(),
getCommonSourceDirectory,
) : undefined,
)));
}
});
}

if (rootNames?.size) {
hostProject.log(`AutoImportProviderProject: found ${rootNames.size} root files in ${dependenciesAdded} dependencies ${referencesAddded} referenced projects in ${timestamp() - start} ms`);
}
return rootNames ? arrayFrom(rootNames.values()) : ts.emptyArray;

function addRootNames(entrypoints: readonly string[] | undefined) {
if (!entrypoints?.length) return 0;
rootNames ??= new Set();
entrypoints.forEach(entry => rootNames!.add(entry));
return 1;
}
return rootNames || ts.emptyArray;

function addDependency(dependency: string) {
if (!startsWith(dependency, "@types/")) {
Expand All @@ -2540,14 +2586,18 @@ export class AutoImportProviderProject extends Project {
});
}

return mapDefined(entrypoints, entrypoint => {
const resolvedFileName = isSymlink ? entrypoint.replace(packageJson.packageDirectory, real!) : entrypoint;
if (!program.getSourceFile(resolvedFileName) && !(isSymlink && program.getSourceFile(entrypoint))) {
return resolvedFileName;
}
});
return filterEntrypoints(entrypoints, isSymlink ? entrypoint => entrypoint.replace(packageJson.packageDirectory, real!) : undefined);
}
}

function filterEntrypoints(entrypoints: readonly string[] | undefined, symlinkName?: (entrypoint: string) => string) {
return mapDefined(entrypoints, entrypoint => {
const resolvedFileName = symlinkName ? symlinkName(entrypoint) : entrypoint;
if (!program!.getSourceFile(resolvedFileName) && !(symlinkName && program!.getSourceFile(entrypoint))) {
return resolvedFileName;
}
});
}
}

/** @internal */
Expand Down
4 changes: 3 additions & 1 deletion src/services/completions.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3948,7 +3948,9 @@ function getCompletionData(
// If module transpilation is enabled or we're targeting es6 or above, or not emitting, OK.
if (compilerOptionsIndicateEsModules(program.getCompilerOptions())) return true;
// If some file is using ES6 modules, assume that it's OK to add more.
return programContainsModules(program);
return program.getSymlinkCache?.().hasAnySymlinks() ||
!!program.getCompilerOptions().paths ||
programContainsModules(program);
}

function isSnippetScope(scopeNode: Node): boolean {
Expand Down
19 changes: 12 additions & 7 deletions src/testRunner/unittests/tsserver/autoImportProvider.ts
Original file line number Diff line number Diff line change
Expand Up @@ -393,6 +393,15 @@ describe("unittests:: tsserver:: autoImportProvider - monorepo", () => {
function setup(files: File[]) {
const host = createServerHost(files);
const session = new TestSession(host);
session.executeCommandSeq<ts.server.protocol.ConfigureRequest>({
command: ts.server.protocol.CommandTypes.Configure,
arguments: {
preferences: {
includePackageJsonAutoImports: "auto",
includeCompletionsForModuleExports: true,
},
},
});
return {
host,
session,
Expand Down Expand Up @@ -427,16 +436,12 @@ function setup(files: File[]) {
}

function triggerCompletions(file: string, line: number, offset: number) {
const requestLocation: ts.server.protocol.FileLocationRequestArgs = {
file,
line,
offset,
};
session.executeCommandSeq<ts.server.protocol.CompletionsRequest>({
command: ts.server.protocol.CommandTypes.CompletionInfo,
arguments: {
...requestLocation,
includeExternalModuleExports: true,
file,
line,
offset,
},
});
}
Expand Down
Loading