2020-12-03 14:42:46 -05:00
|
|
|
/**
|
|
|
|
* @license
|
|
|
|
* Copyright Google LLC All Rights Reserved.
|
|
|
|
*
|
|
|
|
* 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
|
|
|
|
*/
|
|
|
|
|
fix(language-service): show suggestion when type inference is suboptimal (#41072)
The Ivy Language Service uses the compiler's template type-checking engine,
which honors the configuration in the user's tsconfig.json. We recommend
that users upgrade to `strictTemplates` mode in their projects to take
advantage of the best possible type inference, and thus to have the best
experience in Language Service.
If a project is not using `strictTemplates`, then the compiler will not
leverage certain type inference options it has. One case where this is very
noticeable is the inference of let- variables for structural directives that
provide a template context guard (such as NgFor). Without `strictTemplates`,
these guards will not be applied and such variables will be inferred as
'any', degrading the user experience within Language Service.
This is working as designed, since the Language Service _should_ reflect
types exactly as the compiler sees them. However, the View Engine Language
Service used its own type system that _would_ infer these types even when
the compiler did not. As a result, it's confusing to some users why the
Ivy Language Service has "worse" type inference.
To address this confusion, this commit implements a suggestion diagnostic
which is shown in the Language Service for variables which could have been
narrowed via a context guard, but the type checking configuration didn't
allow it. This should make the reason why variables receive the 'any' type
as well as the action needed to improve the typings much more obvious,
improving the Language Service experience.
Fixes angular/vscode-ng-language-service#1155
Closes #41042
PR Close #41072
2021-03-03 18:06:21 -05:00
|
|
|
import {ErrorCode, ngErrorCode} from '@angular/compiler-cli/src/ngtsc/diagnostics';
|
2020-12-03 14:42:46 -05:00
|
|
|
import {initMockFileSystem} from '@angular/compiler-cli/src/ngtsc/file_system/testing';
|
|
|
|
import * as ts from 'typescript';
|
2021-02-08 12:54:28 -05:00
|
|
|
|
|
|
|
import {createModuleAndProjectWithDeclarations, LanguageServiceTestEnv} from '../testing';
|
|
|
|
|
2020-12-03 14:42:46 -05:00
|
|
|
|
|
|
|
describe('getSemanticDiagnostics', () => {
|
2021-02-08 12:54:28 -05:00
|
|
|
let env: LanguageServiceTestEnv;
|
2020-12-03 14:42:46 -05:00
|
|
|
beforeEach(() => {
|
|
|
|
initMockFileSystem('Native');
|
2021-02-08 12:54:28 -05:00
|
|
|
env = LanguageServiceTestEnv.setup();
|
2020-12-03 14:42:46 -05:00
|
|
|
});
|
|
|
|
|
2021-03-02 16:00:45 -05:00
|
|
|
it('should not produce error for a minimal component definition', () => {
|
2021-02-08 12:54:28 -05:00
|
|
|
const files = {
|
|
|
|
'app.ts': `
|
2020-12-03 14:42:46 -05:00
|
|
|
import {Component, NgModule} from '@angular/core';
|
|
|
|
|
|
|
|
@Component({
|
|
|
|
template: ''
|
|
|
|
})
|
|
|
|
export class AppComponent {}
|
|
|
|
`
|
|
|
|
};
|
2021-02-08 12:54:28 -05:00
|
|
|
const project = createModuleAndProjectWithDeclarations(env, 'test', files);
|
2020-12-03 14:42:46 -05:00
|
|
|
|
2021-02-08 12:54:28 -05:00
|
|
|
const diags = project.getDiagnosticsForFile('app.ts');
|
2020-12-03 14:42:46 -05:00
|
|
|
expect(diags.length).toEqual(0);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should report member does not exist', () => {
|
2021-02-08 12:54:28 -05:00
|
|
|
const files = {
|
|
|
|
'app.ts': `
|
2020-12-03 14:42:46 -05:00
|
|
|
import {Component, NgModule} from '@angular/core';
|
|
|
|
|
|
|
|
@Component({
|
|
|
|
template: '{{nope}}'
|
|
|
|
})
|
|
|
|
export class AppComponent {}
|
|
|
|
`
|
|
|
|
};
|
2021-02-08 12:54:28 -05:00
|
|
|
const project = createModuleAndProjectWithDeclarations(env, 'test', files);
|
2020-12-03 14:42:46 -05:00
|
|
|
|
2021-02-08 12:54:28 -05:00
|
|
|
const diags = project.getDiagnosticsForFile('app.ts');
|
2020-12-03 14:42:46 -05:00
|
|
|
expect(diags.length).toBe(1);
|
2021-02-08 12:54:28 -05:00
|
|
|
const {category, file, messageText} = diags[0];
|
2020-12-03 14:42:46 -05:00
|
|
|
expect(category).toBe(ts.DiagnosticCategory.Error);
|
2021-02-08 12:54:28 -05:00
|
|
|
expect(file?.fileName).toBe('/test/app.ts');
|
2020-12-03 14:42:46 -05:00
|
|
|
expect(messageText).toBe(`Property 'nope' does not exist on type 'AppComponent'.`);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should process external template', () => {
|
2021-02-08 12:54:28 -05:00
|
|
|
const files = {
|
|
|
|
'app.ts': `
|
2020-12-03 14:42:46 -05:00
|
|
|
import {Component, NgModule} from '@angular/core';
|
|
|
|
|
|
|
|
@Component({
|
|
|
|
templateUrl: './app.html'
|
|
|
|
})
|
|
|
|
export class AppComponent {}
|
2021-02-08 12:54:28 -05:00
|
|
|
`,
|
|
|
|
'app.html': `Hello world!`
|
2020-12-03 14:42:46 -05:00
|
|
|
};
|
|
|
|
|
2021-02-08 12:54:28 -05:00
|
|
|
const project = createModuleAndProjectWithDeclarations(env, 'test', files);
|
|
|
|
const diags = project.getDiagnosticsForFile('app.html');
|
2020-12-03 14:42:46 -05:00
|
|
|
expect(diags).toEqual([]);
|
|
|
|
});
|
|
|
|
|
2021-03-03 14:54:40 -05:00
|
|
|
it('should not report external template diagnostics on the TS file', () => {
|
|
|
|
const files = {
|
|
|
|
'app.ts': `
|
|
|
|
import {Component, NgModule} from '@angular/core';
|
|
|
|
|
|
|
|
@Component({
|
|
|
|
templateUrl: './app.html'
|
|
|
|
})
|
|
|
|
export class AppComponent {}
|
|
|
|
`,
|
|
|
|
'app.html': '{{nope}}'
|
|
|
|
};
|
|
|
|
|
|
|
|
const project = createModuleAndProjectWithDeclarations(env, 'test', files);
|
|
|
|
const diags = project.getDiagnosticsForFile('app.ts');
|
|
|
|
expect(diags).toEqual([]);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should report diagnostics in inline templates', () => {
|
|
|
|
const files = {
|
|
|
|
'app.ts': `
|
|
|
|
import {Component, NgModule} from '@angular/core';
|
|
|
|
|
|
|
|
@Component({
|
|
|
|
template: '{{nope}}',
|
|
|
|
})
|
|
|
|
export class AppComponent {}
|
|
|
|
`
|
|
|
|
};
|
|
|
|
const project = createModuleAndProjectWithDeclarations(env, 'test', files);
|
|
|
|
const diags = project.getDiagnosticsForFile('app.ts');
|
|
|
|
expect(diags.length).toBe(1);
|
|
|
|
const {category, file, messageText} = diags[0];
|
|
|
|
expect(category).toBe(ts.DiagnosticCategory.Error);
|
|
|
|
expect(file?.fileName).toBe('/test/app.ts');
|
|
|
|
expect(messageText).toBe(`Property 'nope' does not exist on type 'AppComponent'.`);
|
|
|
|
});
|
|
|
|
|
2020-12-03 14:42:46 -05:00
|
|
|
it('should report member does not exist in external template', () => {
|
2021-02-08 12:54:28 -05:00
|
|
|
const files = {
|
|
|
|
'app.ts': `
|
2020-12-03 14:42:46 -05:00
|
|
|
import {Component, NgModule} from '@angular/core';
|
|
|
|
|
|
|
|
@Component({
|
|
|
|
templateUrl: './app.html'
|
|
|
|
})
|
|
|
|
export class AppComponent {}
|
2021-02-08 12:54:28 -05:00
|
|
|
`,
|
|
|
|
'app.html': '{{nope}}'
|
2020-12-03 14:42:46 -05:00
|
|
|
};
|
|
|
|
|
2021-02-08 12:54:28 -05:00
|
|
|
const project = createModuleAndProjectWithDeclarations(env, 'test', files);
|
|
|
|
const diags = project.getDiagnosticsForFile('app.html');
|
2020-12-03 14:42:46 -05:00
|
|
|
expect(diags.length).toBe(1);
|
2021-02-08 12:54:28 -05:00
|
|
|
const {category, file, messageText} = diags[0];
|
2020-12-03 14:42:46 -05:00
|
|
|
expect(category).toBe(ts.DiagnosticCategory.Error);
|
2021-02-08 12:54:28 -05:00
|
|
|
expect(file?.fileName).toBe('/test/app.html');
|
2020-12-03 14:42:46 -05:00
|
|
|
expect(messageText).toBe(`Property 'nope' does not exist on type 'AppComponent'.`);
|
|
|
|
});
|
2020-12-03 14:42:46 -05:00
|
|
|
|
|
|
|
it('should report a parse error in external template', () => {
|
2021-02-08 12:54:28 -05:00
|
|
|
const files = {
|
|
|
|
'app.ts': `
|
2020-12-03 14:42:46 -05:00
|
|
|
import {Component, NgModule} from '@angular/core';
|
|
|
|
|
|
|
|
@Component({
|
|
|
|
templateUrl: './app.html'
|
|
|
|
})
|
|
|
|
export class AppComponent {
|
|
|
|
nope = false;
|
|
|
|
}
|
2021-02-08 12:54:28 -05:00
|
|
|
`,
|
|
|
|
'app.html': '{{nope = true}}'
|
2020-12-03 14:42:46 -05:00
|
|
|
};
|
|
|
|
|
2021-02-08 12:54:28 -05:00
|
|
|
const project = createModuleAndProjectWithDeclarations(env, 'test', files);
|
|
|
|
const diags = project.getDiagnosticsForFile('app.html');
|
2020-12-03 14:42:46 -05:00
|
|
|
expect(diags.length).toBe(1);
|
|
|
|
|
|
|
|
const {category, file, messageText} = diags[0];
|
|
|
|
expect(category).toBe(ts.DiagnosticCategory.Error);
|
2021-02-08 12:54:28 -05:00
|
|
|
expect(file?.fileName).toBe('/test/app.html');
|
2020-12-03 14:42:46 -05:00
|
|
|
expect(messageText)
|
|
|
|
.toContain(
|
|
|
|
`Parser Error: Bindings cannot contain assignments at column 8 in [{{nope = true}}]`);
|
|
|
|
});
|
|
|
|
|
2021-03-02 16:00:45 -05:00
|
|
|
it('reports html parse errors along with typecheck errors as diagnostics', () => {
|
|
|
|
const files = {
|
|
|
|
'app.ts': `
|
|
|
|
import {Component, NgModule} from '@angular/core';
|
|
|
|
|
|
|
|
@Component({
|
|
|
|
templateUrl: './app.html'
|
|
|
|
})
|
|
|
|
export class AppComponent {
|
|
|
|
nope = false;
|
|
|
|
}
|
|
|
|
`,
|
|
|
|
'app.html': '<dne'
|
|
|
|
};
|
|
|
|
|
|
|
|
const project = createModuleAndProjectWithDeclarations(env, 'test', files);
|
|
|
|
const diags = project.getDiagnosticsForFile('app.html');
|
|
|
|
expect(diags.length).toBe(2);
|
|
|
|
|
|
|
|
expect(diags[0].category).toBe(ts.DiagnosticCategory.Error);
|
|
|
|
expect(diags[0].file?.fileName).toBe('/test/app.html');
|
|
|
|
expect(diags[0].messageText).toContain(`'dne' is not a known element`);
|
|
|
|
|
|
|
|
expect(diags[1].category).toBe(ts.DiagnosticCategory.Error);
|
|
|
|
expect(diags[1].file?.fileName).toBe('/test/app.html');
|
|
|
|
expect(diags[1].messageText).toContain(`Opening tag "dne" not terminated.`);
|
|
|
|
});
|
|
|
|
|
2020-12-03 14:42:46 -05:00
|
|
|
it('should report parse errors of components defined in the same ts file', () => {
|
2021-02-08 12:54:28 -05:00
|
|
|
const files = {
|
|
|
|
'app.ts': `
|
2020-12-03 14:42:46 -05:00
|
|
|
import {Component, NgModule} from '@angular/core';
|
|
|
|
|
|
|
|
@Component({ templateUrl: './app1.html' })
|
|
|
|
export class AppComponent1 { nope = false; }
|
|
|
|
|
|
|
|
@Component({ templateUrl: './app2.html' })
|
|
|
|
export class AppComponent2 { nope = false; }
|
2021-02-08 12:54:28 -05:00
|
|
|
`,
|
|
|
|
'app1.html': '{{nope = false}}',
|
|
|
|
'app2.html': '{{nope = true}}',
|
|
|
|
'app-module.ts': `
|
2020-12-03 14:42:46 -05:00
|
|
|
import {NgModule} from '@angular/core';
|
|
|
|
import {CommonModule} from '@angular/common';
|
|
|
|
import {AppComponent, AppComponent2} from './app';
|
|
|
|
|
|
|
|
@NgModule({
|
|
|
|
declarations: [AppComponent, AppComponent2],
|
|
|
|
imports: [CommonModule],
|
|
|
|
})
|
|
|
|
export class AppModule {}
|
2021-02-08 12:54:28 -05:00
|
|
|
`
|
2020-12-03 14:42:46 -05:00
|
|
|
};
|
|
|
|
|
2021-02-08 12:54:28 -05:00
|
|
|
const project = env.addProject('test', files);
|
2021-03-03 14:54:40 -05:00
|
|
|
const diags1 = project.getDiagnosticsForFile('app1.html');
|
|
|
|
expect(diags1.length).toBe(1);
|
|
|
|
expect(diags1[0].messageText)
|
|
|
|
.toBe(
|
|
|
|
'Parser Error: Bindings cannot contain assignments at column 8 in [{{nope = false}}] in /test/app1.html@0:0');
|
|
|
|
|
|
|
|
const diags2 = project.getDiagnosticsForFile('app2.html');
|
|
|
|
expect(diags2.length).toBe(1);
|
|
|
|
expect(diags2[0].messageText)
|
|
|
|
.toBe(
|
|
|
|
'Parser Error: Bindings cannot contain assignments at column 8 in [{{nope = true}}] in /test/app2.html@0:0');
|
2020-12-03 14:42:46 -05:00
|
|
|
});
|
2021-01-06 17:17:45 -05:00
|
|
|
|
|
|
|
it('reports a diagnostic for a component without a template', () => {
|
2021-02-08 12:54:28 -05:00
|
|
|
const files = {
|
|
|
|
'app.ts': `
|
2021-01-06 17:17:45 -05:00
|
|
|
import {Component} from '@angular/core';
|
|
|
|
@Component({})
|
|
|
|
export class MyComponent {}
|
|
|
|
`
|
|
|
|
};
|
|
|
|
|
2021-02-08 12:54:28 -05:00
|
|
|
const project = createModuleAndProjectWithDeclarations(env, 'test', files);
|
|
|
|
const diags = project.getDiagnosticsForFile('app.ts');
|
2021-01-06 17:17:45 -05:00
|
|
|
expect(diags.map(x => x.messageText)).toEqual([
|
|
|
|
'component is missing a template',
|
|
|
|
]);
|
|
|
|
});
|
fix(language-service): show suggestion when type inference is suboptimal (#41072)
The Ivy Language Service uses the compiler's template type-checking engine,
which honors the configuration in the user's tsconfig.json. We recommend
that users upgrade to `strictTemplates` mode in their projects to take
advantage of the best possible type inference, and thus to have the best
experience in Language Service.
If a project is not using `strictTemplates`, then the compiler will not
leverage certain type inference options it has. One case where this is very
noticeable is the inference of let- variables for structural directives that
provide a template context guard (such as NgFor). Without `strictTemplates`,
these guards will not be applied and such variables will be inferred as
'any', degrading the user experience within Language Service.
This is working as designed, since the Language Service _should_ reflect
types exactly as the compiler sees them. However, the View Engine Language
Service used its own type system that _would_ infer these types even when
the compiler did not. As a result, it's confusing to some users why the
Ivy Language Service has "worse" type inference.
To address this confusion, this commit implements a suggestion diagnostic
which is shown in the Language Service for variables which could have been
narrowed via a context guard, but the type checking configuration didn't
allow it. This should make the reason why variables receive the 'any' type
as well as the action needed to improve the typings much more obvious,
improving the Language Service experience.
Fixes angular/vscode-ng-language-service#1155
Closes #41042
PR Close #41072
2021-03-03 18:06:21 -05:00
|
|
|
|
|
|
|
it('reports a warning when the project configuration prevents good type inference', () => {
|
|
|
|
const files = {
|
|
|
|
'app.ts': `
|
|
|
|
import {Component, NgModule} from '@angular/core';
|
|
|
|
import {CommonModule} from '@angular/common';
|
|
|
|
|
|
|
|
@Component({
|
|
|
|
template: '<div *ngFor="let user of users">{{user}}</div>',
|
|
|
|
})
|
|
|
|
export class MyComponent {
|
|
|
|
users = ['Alpha', 'Beta'];
|
|
|
|
}
|
|
|
|
`
|
|
|
|
};
|
|
|
|
|
|
|
|
const project = createModuleAndProjectWithDeclarations(env, 'test', files, {
|
|
|
|
// Disable `strictTemplates`.
|
|
|
|
strictTemplates: false,
|
|
|
|
// Use `fullTemplateTypeCheck` mode instead.
|
|
|
|
fullTemplateTypeCheck: true,
|
|
|
|
});
|
|
|
|
const diags = project.getDiagnosticsForFile('app.ts');
|
|
|
|
expect(diags.length).toBe(1);
|
|
|
|
const diag = diags[0];
|
|
|
|
expect(diag.code).toBe(ngErrorCode(ErrorCode.SUGGEST_SUBOPTIMAL_TYPE_INFERENCE));
|
|
|
|
expect(diag.category).toBe(ts.DiagnosticCategory.Suggestion);
|
|
|
|
expect(getTextOfDiagnostic(diag)).toBe('user');
|
|
|
|
});
|
2021-03-19 10:56:13 -04:00
|
|
|
|
2021-04-08 13:41:32 -04:00
|
|
|
it('should process a component that would otherwise require an inline TCB', () => {
|
|
|
|
const files = {
|
|
|
|
'app.ts': `
|
|
|
|
import {Component, NgModule} from '@angular/core';
|
|
|
|
import {CommonModule} from '@angular/common';
|
|
|
|
|
|
|
|
interface PrivateInterface {}
|
|
|
|
|
|
|
|
@Component({
|
|
|
|
template: 'Simple template',
|
|
|
|
})
|
|
|
|
export class MyComponent<T extends PrivateInterface> {}
|
|
|
|
`
|
|
|
|
};
|
|
|
|
|
|
|
|
const project = createModuleAndProjectWithDeclarations(env, 'test', files);
|
|
|
|
const diags = project.getDiagnosticsForFile('app.ts');
|
|
|
|
expect(diags.length).toBe(0);
|
|
|
|
});
|
|
|
|
|
2021-03-19 10:56:13 -04:00
|
|
|
it('logs perf tracing', () => {
|
|
|
|
const files = {
|
|
|
|
'app.ts': `
|
|
|
|
import {Component} from '@angular/core';
|
|
|
|
@Component({ template: '' })
|
|
|
|
export class MyComponent {}
|
|
|
|
`
|
|
|
|
};
|
|
|
|
|
|
|
|
const project = createModuleAndProjectWithDeclarations(env, 'test', files);
|
|
|
|
|
|
|
|
const logger = project.getLogger();
|
|
|
|
spyOn(logger, 'hasLevel').and.returnValue(true);
|
|
|
|
spyOn(logger, 'perftrc').and.callFake(() => {});
|
|
|
|
|
|
|
|
const diags = project.getDiagnosticsForFile('app.ts');
|
|
|
|
expect(diags.length).toEqual(0);
|
|
|
|
expect(logger.perftrc)
|
|
|
|
.toHaveBeenCalledWith(jasmine.stringMatching(
|
|
|
|
/LanguageService\#LsDiagnostics\:.*\"LsDiagnostics\":\s*\d+.*/g));
|
|
|
|
});
|
2020-12-03 14:42:46 -05:00
|
|
|
});
|
fix(language-service): show suggestion when type inference is suboptimal (#41072)
The Ivy Language Service uses the compiler's template type-checking engine,
which honors the configuration in the user's tsconfig.json. We recommend
that users upgrade to `strictTemplates` mode in their projects to take
advantage of the best possible type inference, and thus to have the best
experience in Language Service.
If a project is not using `strictTemplates`, then the compiler will not
leverage certain type inference options it has. One case where this is very
noticeable is the inference of let- variables for structural directives that
provide a template context guard (such as NgFor). Without `strictTemplates`,
these guards will not be applied and such variables will be inferred as
'any', degrading the user experience within Language Service.
This is working as designed, since the Language Service _should_ reflect
types exactly as the compiler sees them. However, the View Engine Language
Service used its own type system that _would_ infer these types even when
the compiler did not. As a result, it's confusing to some users why the
Ivy Language Service has "worse" type inference.
To address this confusion, this commit implements a suggestion diagnostic
which is shown in the Language Service for variables which could have been
narrowed via a context guard, but the type checking configuration didn't
allow it. This should make the reason why variables receive the 'any' type
as well as the action needed to improve the typings much more obvious,
improving the Language Service experience.
Fixes angular/vscode-ng-language-service#1155
Closes #41042
PR Close #41072
2021-03-03 18:06:21 -05:00
|
|
|
|
|
|
|
function getTextOfDiagnostic(diag: ts.Diagnostic): string {
|
|
|
|
expect(diag.file).not.toBeUndefined();
|
|
|
|
expect(diag.start).not.toBeUndefined();
|
|
|
|
expect(diag.length).not.toBeUndefined();
|
|
|
|
return diag.file!.text.substring(diag.start!, diag.start! + diag.length!);
|
|
|
|
}
|