Skip to content

Commit 4775381

Browse files
authored
Ensure projects are updated for find all references (microsoft#52583)
1 parent 59844e7 commit 4775381

File tree

5 files changed

+670
-2
lines changed

5 files changed

+670
-2
lines changed

src/compiler/program.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -961,14 +961,14 @@ export function createModuleResolutionLoader(
961961
): ResolutionLoader<StringLiteralLike, ResolvedModuleWithFailedLookupLocations, SourceFile> {
962962
return {
963963
nameAndMode: moduleResolutionNameAndModeGetter,
964-
resolve: (moduleName, resoluionMode) => resolveModuleName(
964+
resolve: (moduleName, resolutionMode) => resolveModuleName(
965965
moduleName,
966966
containingFile,
967967
options,
968968
host,
969969
cache,
970970
redirectedReference,
971-
resoluionMode,
971+
resolutionMode,
972972
),
973973
};
974974
}

src/server/session.ts

+6
Original file line numberDiff line numberDiff line change
@@ -730,6 +730,12 @@ function getPerProjectReferences<TResult>(
730730
if (resultsMap.has(project)) continue;
731731
if (isLocationProjectReferenceRedirect(project, location)) continue;
732732

733+
// The project could be dirty and could no longer contain the location's file after it's updated,
734+
// so we need to update the project and check if it still contains the file.
735+
updateProjectIfDirty(project);
736+
if (!project.containsFile(toNormalizedPath(location.fileName))) {
737+
continue;
738+
}
733739
const projectResults = searchPosition(project, location);
734740
resultsMap.set(project, projectResults ?? emptyArray);
735741
searchedProjectKeys.add(getProjectKey(project));

src/testRunner/tests.ts

+1
Original file line numberDiff line numberDiff line change
@@ -53,6 +53,7 @@ import "./unittests/services/extract/constants";
5353
import "./unittests/services/extract/functions";
5454
import "./unittests/services/extract/symbolWalker";
5555
import "./unittests/services/extract/ranges";
56+
import "./unittests/services/findAllReferences";
5657
import "./unittests/services/hostNewLineSupport";
5758
import "./unittests/services/languageService";
5859
import "./unittests/services/organizeImports";
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,141 @@
1+
import { protocol } from "../../_namespaces/ts.server";
2+
import { baselineTsserverLogs, createLoggerWithInMemoryLogs, createSession } from "../tsserver/helpers";
3+
import { createServerHost, File } from "../virtualFileSystemWithWatch";
4+
5+
describe("unittests:: services:: findAllReferences", () => {
6+
it("does not try to open a file in a project that was updated and no longer has the file", () => {
7+
const files: File[] = [
8+
{
9+
path: "/packages/babel-loader/tsconfig.json",
10+
content:
11+
`
12+
{
13+
"compilerOptions": {
14+
"target": "ES2018",
15+
"module": "commonjs",
16+
"strict": true,
17+
"esModuleInterop": true,
18+
"composite": true,
19+
"rootDir": "src",
20+
"outDir": "dist"
21+
},
22+
"include": ["src"],
23+
"references": [{"path": "../core"}]
24+
}
25+
`
26+
},
27+
{
28+
path: "/packages/babel-loader/src/index.ts",
29+
content:
30+
`
31+
import type { Foo } from "../../core/src/index.js";
32+
`
33+
},
34+
{
35+
path: "/packages/core/tsconfig.json",
36+
content:
37+
`
38+
{
39+
"compilerOptions": {
40+
"target": "ES2018",
41+
"module": "commonjs",
42+
"strict": true,
43+
"esModuleInterop": true,
44+
"composite": true,
45+
"rootDir": "./src",
46+
"outDir": "./dist",
47+
},
48+
"include": ["./src"]
49+
}
50+
`
51+
},
52+
{
53+
path: "/packages/core/src/index.ts",
54+
content:
55+
`
56+
import { Bar } from "./loading-indicator.js";
57+
export type Foo = {};
58+
const bar: Bar = {
59+
prop: 0
60+
}
61+
`
62+
},
63+
{
64+
path: "/packages/core/src/loading-indicator.ts",
65+
content:
66+
`
67+
export interface Bar {
68+
prop: number;
69+
}
70+
const bar: Bar = {
71+
prop: 1
72+
}
73+
`
74+
},
75+
];
76+
const host = createServerHost(files);
77+
const session = createSession(host, { logger: createLoggerWithInMemoryLogs(host) });
78+
// Open files in the two configured projects
79+
session.executeCommandSeq<protocol.UpdateOpenRequest>({
80+
command: protocol.CommandTypes.UpdateOpen,
81+
arguments: {
82+
openFiles: [
83+
{
84+
file: files[1].path, // babel-loader/src/index.ts
85+
fileContent: files[1].content,
86+
}
87+
]
88+
}
89+
});
90+
session.executeCommandSeq<protocol.UpdateOpenRequest>({
91+
command: protocol.CommandTypes.UpdateOpen,
92+
arguments: {
93+
openFiles: [
94+
{
95+
file: files[3].path, // core/src/index.ts
96+
fileContent: files[3].content,
97+
}
98+
]
99+
}
100+
});
101+
// Now change `babel-loader` project to no longer import `core` project
102+
session.executeCommandSeq<protocol.UpdateOpenRequest>({
103+
command: protocol.CommandTypes.UpdateOpen,
104+
arguments: {
105+
changedFiles: [
106+
{
107+
fileName: files[1].path,
108+
textChanges: [
109+
{
110+
start: {
111+
line: 1,
112+
offset: 26
113+
},
114+
end: {
115+
line: 1,
116+
offset: 26
117+
},
118+
newText: "// comment",
119+
}
120+
]
121+
}
122+
]
123+
}
124+
});
125+
const loadingIndicatorScriptInfo = session.getProjectService().getScriptInfo(files[3].path)!;
126+
// At this point, we haven't updated `babel-loader` project yet,
127+
// so `babel-loader` is still a containing project of `loading-indicator` file.
128+
assert(loadingIndicatorScriptInfo.containingProjects.find(p => p.projectName === "/packages/babel-loader/tsconfig.json"));
129+
// When calling find all references,
130+
// we shouldn't crash due to using outdated information on a file's containig projects.
131+
session.executeCommandSeq<protocol.ReferencesRequest>({
132+
command: protocol.CommandTypes.References,
133+
arguments: {
134+
file: files[3].path, // core/src/index.ts
135+
line: 5, // `prop`
136+
offset: 5,
137+
}
138+
});
139+
baselineTsserverLogs("findAllReferences", "does not try to open a file in a project that was updated and no longer has the file", session);
140+
});
141+
});

0 commit comments

Comments
 (0)