2020-04-29 18:52:17 -04:00
|
|
|
/**
|
|
|
|
* @license
|
2020-05-19 15:08:49 -04:00
|
|
|
* Copyright Google LLC All Rights Reserved.
|
2020-04-29 18:52:17 -04:00
|
|
|
*
|
|
|
|
* Use of this source code is governed by an MIT-style license that can be
|
|
|
|
* found in the LICENSE file at https://angular.io/license
|
|
|
|
*/
|
|
|
|
|
2021-01-13 17:52:29 -05:00
|
|
|
import {ErrorCode, ngErrorCode} from '@angular/compiler-cli/src/ngtsc/diagnostics';
|
2020-10-16 14:21:24 -04:00
|
|
|
import * as ts from 'typescript/lib/tsserverlibrary';
|
2020-10-09 19:46:55 -04:00
|
|
|
|
2020-11-09 19:20:02 -05:00
|
|
|
import {LanguageService} from '../../language_service';
|
2020-10-09 19:46:55 -04:00
|
|
|
|
2020-11-09 19:20:02 -05:00
|
|
|
import {MockConfigFileFs, MockService, setup, TEST_TEMPLATE, TSCONFIG} from './mock_host';
|
2020-10-09 19:46:55 -04:00
|
|
|
|
2020-10-16 14:21:24 -04:00
|
|
|
describe('language service adapter', () => {
|
|
|
|
let project: ts.server.Project;
|
|
|
|
let service: MockService;
|
|
|
|
let ngLS: LanguageService;
|
2020-11-09 19:20:02 -05:00
|
|
|
let configFileFs: MockConfigFileFs;
|
2020-10-09 19:46:55 -04:00
|
|
|
|
2020-10-16 14:21:24 -04:00
|
|
|
beforeAll(() => {
|
2020-11-09 19:20:02 -05:00
|
|
|
const {project: _project, tsLS, service: _service, configFileFs: _configFileFs} = setup();
|
2020-10-16 14:21:24 -04:00
|
|
|
project = _project;
|
|
|
|
service = _service;
|
2021-03-02 18:46:11 -05:00
|
|
|
ngLS = new LanguageService(project, tsLS, {});
|
2020-11-09 19:20:02 -05:00
|
|
|
configFileFs = _configFileFs;
|
2020-10-09 19:46:55 -04:00
|
|
|
});
|
|
|
|
|
2020-11-09 19:20:02 -05:00
|
|
|
afterEach(() => {
|
|
|
|
configFileFs.clear();
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('parse compiler options', () => {
|
|
|
|
it('should initialize with angularCompilerOptions from tsconfig.json', () => {
|
|
|
|
expect(ngLS.getCompilerOptions()).toEqual(jasmine.objectContaining({
|
2020-10-16 14:21:24 -04:00
|
|
|
enableIvy: true, // default for ivy is true
|
|
|
|
strictTemplates: true,
|
|
|
|
strictInjectionParameters: true,
|
|
|
|
}));
|
|
|
|
});
|
2020-11-09 19:20:02 -05:00
|
|
|
|
|
|
|
it('should reparse angularCompilerOptions on tsconfig.json change', () => {
|
|
|
|
expect(ngLS.getCompilerOptions()).toEqual(jasmine.objectContaining({
|
|
|
|
enableIvy: true, // default for ivy is true
|
|
|
|
strictTemplates: true,
|
|
|
|
strictInjectionParameters: true,
|
|
|
|
}));
|
|
|
|
|
2021-01-13 17:52:29 -05:00
|
|
|
configFileFs.overwriteConfigFile(TSCONFIG, {
|
|
|
|
angularCompilerOptions: {
|
|
|
|
strictTemplates: false,
|
|
|
|
}
|
|
|
|
});
|
2020-11-09 19:20:02 -05:00
|
|
|
|
|
|
|
expect(ngLS.getCompilerOptions()).toEqual(jasmine.objectContaining({
|
|
|
|
strictTemplates: false,
|
|
|
|
}));
|
|
|
|
});
|
2021-03-02 18:46:11 -05:00
|
|
|
|
|
|
|
it('should always enable strictTemplates if forceStrictTemplates is true', () => {
|
|
|
|
const {project, tsLS, configFileFs} = setup();
|
|
|
|
const ngLS = new LanguageService(project, tsLS, {
|
|
|
|
forceStrictTemplates: true,
|
|
|
|
});
|
|
|
|
|
|
|
|
// First make sure the default for strictTemplates is true
|
|
|
|
expect(ngLS.getCompilerOptions()).toEqual(jasmine.objectContaining({
|
|
|
|
enableIvy: true, // default for ivy is true
|
|
|
|
strictTemplates: true,
|
|
|
|
strictInjectionParameters: true,
|
|
|
|
}));
|
|
|
|
|
|
|
|
// Change strictTemplates to false
|
|
|
|
configFileFs.overwriteConfigFile(TSCONFIG, {
|
|
|
|
angularCompilerOptions: {
|
|
|
|
strictTemplates: false,
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
// Make sure strictTemplates is still true because forceStrictTemplates
|
|
|
|
// is enabled.
|
|
|
|
expect(ngLS.getCompilerOptions()).toEqual(jasmine.objectContaining({
|
|
|
|
strictTemplates: true,
|
|
|
|
}));
|
|
|
|
});
|
2020-10-09 19:46:55 -04:00
|
|
|
});
|
|
|
|
|
2021-01-13 17:52:29 -05:00
|
|
|
describe('compiler options diagnostics', () => {
|
|
|
|
it('suggests turning on strict flag', () => {
|
|
|
|
configFileFs.overwriteConfigFile(TSCONFIG, {
|
|
|
|
angularCompilerOptions: {},
|
|
|
|
});
|
|
|
|
const diags = ngLS.getCompilerOptionsDiagnostics();
|
|
|
|
const diag = diags.find(isSuggestStrictTemplatesDiag);
|
|
|
|
expect(diag).toBeDefined();
|
|
|
|
expect(diag!.category).toBe(ts.DiagnosticCategory.Suggestion);
|
|
|
|
expect(diag!.file?.getSourceFile().fileName).toBe(TSCONFIG);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('does not suggest turning on strict mode is strictTemplates flag is on', () => {
|
|
|
|
configFileFs.overwriteConfigFile(TSCONFIG, {
|
|
|
|
angularCompilerOptions: {
|
|
|
|
strictTemplates: true,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
const diags = ngLS.getCompilerOptionsDiagnostics();
|
|
|
|
const diag = diags.find(isSuggestStrictTemplatesDiag);
|
|
|
|
expect(diag).toBeUndefined();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('does not suggest turning on strict mode is fullTemplateTypeCheck flag is on', () => {
|
|
|
|
configFileFs.overwriteConfigFile(TSCONFIG, {
|
|
|
|
angularCompilerOptions: {
|
|
|
|
fullTemplateTypeCheck: true,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
const diags = ngLS.getCompilerOptionsDiagnostics();
|
|
|
|
const diag = diags.find(isSuggestStrictTemplatesDiag);
|
|
|
|
expect(diag).toBeUndefined();
|
|
|
|
});
|
|
|
|
|
|
|
|
function isSuggestStrictTemplatesDiag(diag: ts.Diagnostic) {
|
|
|
|
return diag.code === ngErrorCode(ErrorCode.SUGGEST_STRICT_TEMPLATES);
|
|
|
|
}
|
|
|
|
});
|
2020-11-09 19:20:02 -05:00
|
|
|
|
2020-10-16 14:21:24 -04:00
|
|
|
describe('last known program', () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
service.reset();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should be set after getSemanticDiagnostics()', () => {
|
|
|
|
const d0 = ngLS.getSemanticDiagnostics(TEST_TEMPLATE);
|
|
|
|
expect(d0.length).toBe(0);
|
|
|
|
const p0 = getLastKnownProgram(ngLS);
|
|
|
|
|
|
|
|
const d1 = ngLS.getSemanticDiagnostics(TEST_TEMPLATE);
|
|
|
|
expect(d1.length).toBe(0);
|
|
|
|
const p1 = getLastKnownProgram(ngLS);
|
|
|
|
expect(p1).toBe(p0); // last known program should not have changed
|
|
|
|
|
|
|
|
service.overwrite(TEST_TEMPLATE, `<test-c¦omp></test-comp>`);
|
|
|
|
const d2 = ngLS.getSemanticDiagnostics(TEST_TEMPLATE);
|
|
|
|
expect(d2.length).toBe(0);
|
|
|
|
const p2 = getLastKnownProgram(ngLS);
|
|
|
|
expect(p2).not.toBe(p1); // last known program should have changed
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should be set after getDefinitionAndBoundSpan()', () => {
|
|
|
|
const {position: pos0} = service.overwrite(TEST_TEMPLATE, `<test-c¦omp></test-comp>`);
|
|
|
|
|
|
|
|
const d0 = ngLS.getDefinitionAndBoundSpan(TEST_TEMPLATE, pos0);
|
|
|
|
expect(d0).toBeDefined();
|
|
|
|
const p0 = getLastKnownProgram(ngLS);
|
|
|
|
|
|
|
|
const d1 = ngLS.getDefinitionAndBoundSpan(TEST_TEMPLATE, pos0);
|
|
|
|
expect(d1).toBeDefined();
|
|
|
|
const p1 = getLastKnownProgram(ngLS);
|
|
|
|
expect(p1).toBe(p0); // last known program should not have changed
|
|
|
|
|
|
|
|
const {position: pos1} = service.overwrite(TEST_TEMPLATE, `{{ ti¦tle }}`);
|
|
|
|
const d2 = ngLS.getDefinitionAndBoundSpan(TEST_TEMPLATE, pos1);
|
|
|
|
expect(d2).toBeDefined();
|
|
|
|
const p2 = getLastKnownProgram(ngLS);
|
|
|
|
expect(p2).not.toBe(p1); // last known program should have changed
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should be set after getQuickInfoAtPosition()', () => {
|
|
|
|
const {position: pos0} = service.overwrite(TEST_TEMPLATE, `<test-c¦omp></test-comp>`);
|
|
|
|
|
|
|
|
const q0 = ngLS.getQuickInfoAtPosition(TEST_TEMPLATE, pos0);
|
|
|
|
expect(q0).toBeDefined();
|
|
|
|
const p0 = getLastKnownProgram(ngLS);
|
|
|
|
|
|
|
|
const q1 = ngLS.getQuickInfoAtPosition(TEST_TEMPLATE, pos0);
|
|
|
|
expect(q1).toBeDefined();
|
|
|
|
const p1 = getLastKnownProgram(ngLS);
|
|
|
|
expect(p1).toBe(p0); // last known program should not have changed
|
|
|
|
|
|
|
|
const {position: pos1} = service.overwrite(TEST_TEMPLATE, `{{ ti¦tle }}`);
|
|
|
|
const q2 = ngLS.getQuickInfoAtPosition(TEST_TEMPLATE, pos1);
|
|
|
|
expect(q2).toBeDefined();
|
|
|
|
const p2 = getLastKnownProgram(ngLS);
|
|
|
|
expect(p2).not.toBe(p1); // last known program should have changed
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should be set after getTypeDefinitionAtPosition()', () => {
|
|
|
|
const {position: pos0} = service.overwrite(TEST_TEMPLATE, `<test-c¦omp></test-comp>`);
|
|
|
|
|
|
|
|
const q0 = ngLS.getTypeDefinitionAtPosition(TEST_TEMPLATE, pos0);
|
|
|
|
expect(q0).toBeDefined();
|
|
|
|
const p0 = getLastKnownProgram(ngLS);
|
|
|
|
|
|
|
|
const d1 = ngLS.getTypeDefinitionAtPosition(TEST_TEMPLATE, pos0);
|
|
|
|
expect(d1).toBeDefined();
|
|
|
|
const p1 = getLastKnownProgram(ngLS);
|
|
|
|
expect(p1).toBe(p0); // last known program should not have changed
|
|
|
|
|
|
|
|
const {position: pos1} = service.overwrite(TEST_TEMPLATE, `{{ ti¦tle }}`);
|
|
|
|
const d2 = ngLS.getTypeDefinitionAtPosition(TEST_TEMPLATE, pos1);
|
|
|
|
expect(d2).toBeDefined();
|
|
|
|
const p2 = getLastKnownProgram(ngLS);
|
|
|
|
expect(p2).not.toBe(p1); // last known program should have changed
|
|
|
|
});
|
2020-10-09 19:46:55 -04:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
function getLastKnownProgram(ngLS: LanguageService): ts.Program {
|
2021-04-08 16:14:45 -04:00
|
|
|
const program = ngLS['compilerFactory']['compiler']?.getCurrentProgram();
|
2020-10-09 19:46:55 -04:00
|
|
|
expect(program).toBeDefined();
|
|
|
|
return program!;
|
|
|
|
}
|