Skip to content

Commit 4f1806e

Browse files
authored
fix(typescript-estree): allow providing more one than one existing program in config (typescript-eslint#3508)
1 parent ced9b26 commit 4f1806e

File tree

10 files changed

+86
-49
lines changed

10 files changed

+86
-49
lines changed

.eslintrc.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -27,7 +27,7 @@ module.exports = {
2727
],
2828
tsconfigRootDir: __dirname,
2929
warnOnUnsupportedTypeScriptVersion: false,
30-
EXPERIMENTAL_useSourceOfProjectReferenceRedirect: true,
30+
EXPERIMENTAL_useSourceOfProjectReferenceRedirect: false,
3131
},
3232
rules: {
3333
//

packages/experimental-utils/src/ast-utils/eslint-utils/ReferenceTracker.ts

-2
Original file line numberDiff line numberDiff line change
@@ -3,12 +3,10 @@ import * as eslintUtils from 'eslint-utils';
33
import { TSESTree } from '../../ts-estree';
44
import * as TSESLint from '../../ts-eslint';
55

6-
/* eslint-disable @typescript-eslint/no-unsafe-assignment */
76
const ReferenceTrackerREAD: unique symbol = eslintUtils.ReferenceTracker.READ;
87
const ReferenceTrackerCALL: unique symbol = eslintUtils.ReferenceTracker.CALL;
98
const ReferenceTrackerCONSTRUCT: unique symbol =
109
eslintUtils.ReferenceTracker.CONSTRUCT;
11-
/* eslint-enable @typescript-eslint/no-unsafe-assignment */
1210

1311
interface ReferenceTracker {
1412
/**

packages/experimental-utils/src/ts-eslint-scope/index.ts

-1
Original file line numberDiff line numberDiff line change
@@ -10,5 +10,4 @@ export * from './Scope';
1010
export * from './ScopeManager';
1111
export * from './Variable';
1212

13-
// eslint-disable-next-line @typescript-eslint/no-unsafe-assignment
1413
export const version: string = ESLintVersion;

packages/parser/README.md

+7-7
Original file line numberDiff line numberDiff line change
@@ -213,19 +213,19 @@ Default `false`.
213213

214214
This option allows you to request that when the `project` setting is specified, files will be allowed when not included in the projects defined by the provided `tsconfig.json` files. **Using this option will incur significant performance costs. This option is primarily included for backwards-compatibility.** See the **`project`** section above for more information.
215215

216-
### `parserOptions.program`
216+
### `parserOptions.programs`
217217

218218
Default `undefined`.
219219

220-
This option allows you to programmatically provide an instance of a TypeScript Program object that will provide type information to rules.
221-
This will override any program that would have been computed from `parserOptions.project` or `parserOptions.createDefaultProgram`.
222-
All linted files must be part of the provided program.
220+
This option allows you to programmatically provide an array of one or more instances of a TypeScript Program object that will provide type information to rules.
221+
This will override any programs that would have been computed from `parserOptions.project` or `parserOptions.createDefaultProgram`.
222+
All linted files must be part of the provided program(s).
223223

224224
## Utilities
225225

226226
### `createProgram(configFile, projectDirectory)`
227227

228-
This serves as a utility method for users of the `parserOptions.program` feature to create a TypeScript program instance from a config file.
228+
This serves as a utility method for users of the `parserOptions.programs` feature to create a TypeScript program instance from a config file.
229229

230230
```ts
231231
declare function createProgram(
@@ -238,10 +238,10 @@ Example usage in .eslintrc.js:
238238

239239
```js
240240
const parser = require('@typescript-eslint/parser');
241-
const program = parser.createProgram('tsconfig.json');
241+
const programs = [parser.createProgram('tsconfig.json')];
242242
module.exports = {
243243
parserOptions: {
244-
program,
244+
programs,
245245
},
246246
};
247247
```

packages/typescript-estree/src/create-program/useProvidedProgram.ts packages/typescript-estree/src/create-program/useProvidedPrograms.ts

+19-9
Original file line numberDiff line numberDiff line change
@@ -11,29 +11,39 @@ import {
1111

1212
const log = debug('typescript-eslint:typescript-estree:useProvidedProgram');
1313

14-
function useProvidedProgram(
15-
programInstance: ts.Program,
14+
function useProvidedPrograms(
15+
programInstances: ts.Program[],
1616
extra: Extra,
1717
): ASTAndProgram | undefined {
18-
log('Retrieving ast for %s from provided program instance', extra.filePath);
19-
20-
programInstance.getTypeChecker(); // ensure parent pointers are set in source files
18+
log(
19+
'Retrieving ast for %s from provided program instance(s)',
20+
extra.filePath,
21+
);
2122

22-
const astAndProgram = getAstFromProgram(programInstance, extra);
23+
let astAndProgram: ASTAndProgram | undefined;
24+
for (const programInstance of programInstances) {
25+
astAndProgram = getAstFromProgram(programInstance, extra);
26+
// Stop at the first applicable program instance
27+
if (astAndProgram) {
28+
break;
29+
}
30+
}
2331

2432
if (!astAndProgram) {
2533
const relativeFilePath = path.relative(
2634
extra.tsconfigRootDir || process.cwd(),
2735
extra.filePath,
2836
);
2937
const errorLines = [
30-
'"parserOptions.program" has been provided for @typescript-eslint/parser.',
31-
`The file was not found in the provided program instance: ${relativeFilePath}`,
38+
'"parserOptions.programs" has been provided for @typescript-eslint/parser.',
39+
`The file was not found in any of the provided program instance(s): ${relativeFilePath}`,
3240
];
3341

3442
throw new Error(errorLines.join('\n'));
3543
}
3644

45+
astAndProgram.program.getTypeChecker(); // ensure parent pointers are set in source files
46+
3747
return astAndProgram;
3848
}
3949

@@ -84,4 +94,4 @@ function formatDiagnostics(diagnostics: ts.Diagnostic[]): string | undefined {
8494
});
8595
}
8696

87-
export { useProvidedProgram, createProgramFromConfigFile };
97+
export { useProvidedPrograms, createProgramFromConfigFile };

packages/typescript-estree/src/index.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@ export { ParserServices, TSESTreeOptions } from './parser-options';
33
export { simpleTraverse } from './simple-traverse';
44
export * from './ts-estree';
55
export { clearCaches } from './create-program/createWatchProgram';
6-
export { createProgramFromConfigFile as createProgram } from './create-program/useProvidedProgram';
6+
export { createProgramFromConfigFile as createProgram } from './create-program/useProvidedPrograms';
77

88
// re-export for backwards-compat
99
export { visitorKeys } from '@typescript-eslint/visitor-keys';

packages/typescript-estree/src/parser-options.ts

+4-4
Original file line numberDiff line numberDiff line change
@@ -20,7 +20,7 @@ export interface Extra {
2020
loc: boolean;
2121
log: (message: string) => void;
2222
preserveNodeMaps?: boolean;
23-
program: null | Program;
23+
programs: null | Program[];
2424
projects: CanonicalPath[];
2525
range: boolean;
2626
strict: boolean;
@@ -171,11 +171,11 @@ interface ParseAndGenerateServicesOptions extends ParseOptions {
171171
tsconfigRootDir?: string;
172172

173173
/**
174-
* Instance of a TypeScript Program object to be used for type information.
174+
* An array of one or more instances of TypeScript Program objects to be used for type information.
175175
* This overrides any program or programs that would have been computed from the `project` option.
176-
* All linted files must be part of the provided program.
176+
* All linted files must be part of the provided program(s).
177177
*/
178-
program?: Program;
178+
programs?: Program[];
179179

180180
/**
181181
***************************************************************************************

packages/typescript-estree/src/parser.ts

+17-10
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,7 @@ import {
1919
getCanonicalFileName,
2020
} from './create-program/shared';
2121
import { Program } from 'typescript';
22-
import { useProvidedProgram } from './create-program/useProvidedProgram';
22+
import { useProvidedPrograms } from './create-program/useProvidedPrograms';
2323

2424
const log = debug('typescript-eslint:typescript-estree:parser');
2525

@@ -57,18 +57,20 @@ function enforceString(code: unknown): string {
5757

5858
/**
5959
* @param code The code of the file being linted
60+
* @param programInstances One or more existing programs to use
6061
* @param shouldProvideParserServices True if the program should be attempted to be calculated from provided tsconfig files
6162
* @param shouldCreateDefaultProgram True if the program should be created from compiler host
6263
* @returns Returns a source file and program corresponding to the linted code
6364
*/
6465
function getProgramAndAST(
6566
code: string,
66-
programInstance: Program | null,
67+
programInstances: Program[] | null,
6768
shouldProvideParserServices: boolean,
6869
shouldCreateDefaultProgram: boolean,
6970
): ASTAndProgram {
7071
return (
71-
(programInstance && useProvidedProgram(programInstance, extra)) ||
72+
(programInstances?.length &&
73+
useProvidedPrograms(programInstances, extra)) ||
7274
(shouldProvideParserServices &&
7375
createProjectProgram(code, shouldCreateDefaultProgram, extra)) ||
7476
(shouldProvideParserServices &&
@@ -109,7 +111,7 @@ function resetExtra(): void {
109111
loc: false,
110112
log: console.log, // eslint-disable-line no-console
111113
preserveNodeMaps: true,
112-
program: null,
114+
programs: null,
113115
projects: [],
114116
range: false,
115117
strict: false,
@@ -269,14 +271,19 @@ function applyParserOptionsToExtra(options: TSESTreeOptions): void {
269271
// NOTE - ensureAbsolutePath relies upon having the correct tsconfigRootDir in extra
270272
extra.filePath = ensureAbsolutePath(extra.filePath, extra);
271273

272-
if (options.program && typeof options.program === 'object') {
273-
extra.program = options.program;
274+
if (Array.isArray(options.programs)) {
275+
if (!options.programs.length) {
276+
throw new Error(
277+
`You have set parserOptions.programs to an empty array. This will cause all files to not be found in existing programs. Either provide one or more existing TypeScript Program instances in the array, or remove the parserOptions.programs setting.`,
278+
);
279+
}
280+
extra.programs = options.programs;
274281
log(
275-
'parserOptions.program was provided, so parserOptions.project will be ignored.',
282+
'parserOptions.programs was provided, so parserOptions.project will be ignored.',
276283
);
277284
}
278285

279-
if (!extra.program) {
286+
if (!extra.programs) {
280287
// providing a program overrides project resolution
281288
const projectFolderIgnoreList = (
282289
options.projectFolderIgnoreList ?? ['**/node_modules/**']
@@ -464,10 +471,10 @@ function parseAndGenerateServices<T extends TSESTreeOptions = TSESTreeOptions>(
464471
* Generate a full ts.Program or offer provided instance in order to be able to provide parser services, such as type-checking
465472
*/
466473
const shouldProvideParserServices =
467-
extra.program != null || (extra.projects && extra.projects.length > 0);
474+
extra.programs != null || (extra.projects && extra.projects.length > 0);
468475
const { ast, program } = getProgramAndAST(
469476
code,
470-
extra.program,
477+
extra.programs,
471478
shouldProvideParserServices,
472479
extra.createDefaultProgram,
473480
)!;

packages/typescript-estree/tests/lib/semanticInfo.test.ts

+34-13
Original file line numberDiff line numberDiff line change
@@ -293,37 +293,58 @@ describe('semanticInfo', () => {
293293
expect(parseResult.services.program).toBeDefined();
294294
});
295295

296-
it(`provided program instance is returned in result`, () => {
296+
it('empty programs array should throw', () => {
297+
const fileName = path.resolve(FIXTURES_DIR, 'isolated-file.src.ts');
298+
const badConfig = createOptions(fileName);
299+
badConfig.programs = [];
300+
expect(() => parseAndGenerateServices('const foo = 5;', badConfig)).toThrow(
301+
'You have set parserOptions.programs to an empty array. This will cause all files to not be found in existing programs. Either provide one or more existing TypeScript Program instances in the array, or remove the parserOptions.programs setting.',
302+
);
303+
});
304+
305+
it(`first matching provided program instance is returned in result`, () => {
297306
const filename = testFiles[0];
298-
const program = createProgram(path.join(FIXTURES_DIR, 'tsconfig.json'));
307+
const program1 = createProgram(path.join(FIXTURES_DIR, 'tsconfig.json'));
308+
const program2 = createProgram(path.join(FIXTURES_DIR, 'tsconfig.json'));
299309
const code = fs.readFileSync(path.join(FIXTURES_DIR, filename), 'utf8');
300310
const options = createOptions(filename);
301311
const optionsProjectString = {
302312
...options,
303-
program: program,
313+
programs: [program1, program2],
304314
project: './tsconfig.json',
305315
};
306316
const parseResult = parseAndGenerateServices(code, optionsProjectString);
307-
expect(parseResult.services.program).toBe(program);
317+
expect(parseResult.services.program).toBe(program1);
308318
});
309319

310-
it('file not in provided program instance', () => {
311-
const filename = 'non-existant-file.ts';
312-
const program = createProgram(path.join(FIXTURES_DIR, 'tsconfig.json'));
320+
it('file not in provided program instance(s)', () => {
321+
const filename = 'non-existent-file.ts';
322+
const program1 = createProgram(path.join(FIXTURES_DIR, 'tsconfig.json'));
313323
const options = createOptions(filename);
314-
const optionsProjectString = {
324+
const optionsWithSingleProgram = {
325+
...options,
326+
programs: [program1],
327+
};
328+
expect(() =>
329+
parseAndGenerateServices('const foo = 5;', optionsWithSingleProgram),
330+
).toThrow(
331+
`The file was not found in any of the provided program instance(s): ${filename}`,
332+
);
333+
334+
const program2 = createProgram(path.join(FIXTURES_DIR, 'tsconfig.json'));
335+
const optionsWithMultiplePrograms = {
315336
...options,
316-
program: program,
337+
programs: [program1, program2],
317338
};
318339
expect(() =>
319-
parseAndGenerateServices('const foo = 5;', optionsProjectString),
340+
parseAndGenerateServices('const foo = 5;', optionsWithMultiplePrograms),
320341
).toThrow(
321-
`The file was not found in the provided program instance: ${filename}`,
342+
`The file was not found in any of the provided program instance(s): ${filename}`,
322343
);
323344
});
324345

325-
it('createProgram fails on non-existant file', () => {
326-
expect(() => createProgram('tsconfig.non-existant.json')).toThrow();
346+
it('createProgram fails on non-existent file', () => {
347+
expect(() => createProgram('tsconfig.non-existent.json')).toThrow();
327348
});
328349

329350
it('createProgram fails on tsconfig with errors', () => {

tsconfig.eslint.json

+3-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,8 @@
11
{
22
"compilerOptions": {
3-
"types": ["@types/node"]
3+
"types": ["@types/node"],
4+
"noEmit": true,
5+
"allowJs": true
46
},
57
"extends": "./tsconfig.base.json",
68
"include": ["tests/**/*.ts", "tools/**/*.ts", ".eslintrc.js"]

0 commit comments

Comments
 (0)