2016-11-22 12:10:23 -05:00
|
|
|
/**
|
|
|
|
* @license
|
|
|
|
* Copyright Google Inc. 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
|
|
|
|
*/
|
|
|
|
|
|
|
|
import * as ts from 'typescript';
|
|
|
|
|
2019-12-04 15:02:29 -05:00
|
|
|
import {create} from '../src/ts_plugin';
|
2019-10-14 18:24:28 -04:00
|
|
|
import {CompletionKind} from '../src/types';
|
2016-11-22 12:10:23 -05:00
|
|
|
|
2019-10-14 18:24:28 -04:00
|
|
|
import {MockTypescriptHost} from './test_utils';
|
2016-11-22 12:10:23 -05:00
|
|
|
|
2019-10-14 18:24:28 -04:00
|
|
|
const mockProject = {
|
|
|
|
projectService: {
|
|
|
|
logger: {
|
|
|
|
info() {},
|
|
|
|
hasLevel: () => false,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
hasRoots: () => true,
|
|
|
|
} as any;
|
2016-11-22 12:10:23 -05:00
|
|
|
|
2019-10-14 18:24:28 -04:00
|
|
|
describe('plugin', () => {
|
|
|
|
const mockHost = new MockTypescriptHost(['/app/main.ts']);
|
|
|
|
const tsLS = ts.createLanguageService(mockHost);
|
2020-04-03 23:57:39 -04:00
|
|
|
const program = tsLS.getProgram()!;
|
2019-10-14 18:24:28 -04:00
|
|
|
const plugin = create({
|
|
|
|
languageService: tsLS,
|
|
|
|
languageServiceHost: mockHost,
|
|
|
|
project: mockProject,
|
|
|
|
serverHost: {} as any,
|
|
|
|
config: {},
|
2016-11-22 12:10:23 -05:00
|
|
|
});
|
|
|
|
|
2020-04-03 23:57:39 -04:00
|
|
|
beforeEach(() => {
|
|
|
|
mockHost.reset();
|
|
|
|
});
|
2016-11-22 12:10:23 -05:00
|
|
|
|
2019-10-14 18:24:28 -04:00
|
|
|
it('should produce TypeScript diagnostics', () => {
|
|
|
|
const fileName = '/foo.ts';
|
|
|
|
mockHost.addScript(fileName, `
|
|
|
|
function add(x: number) {
|
|
|
|
return x + 42;
|
|
|
|
}
|
|
|
|
add('hello');
|
|
|
|
`);
|
|
|
|
const diags = plugin.getSemanticDiagnostics(fileName);
|
|
|
|
expect(diags.length).toBe(1);
|
|
|
|
expect(diags[0].messageText)
|
|
|
|
.toBe(`Argument of type '"hello"' is not assignable to parameter of type 'number'.`);
|
2019-09-12 18:20:54 -04:00
|
|
|
});
|
2016-11-22 12:10:23 -05:00
|
|
|
|
2019-10-14 18:24:28 -04:00
|
|
|
it('should not report TypeScript errors on tour of heroes', () => {
|
|
|
|
const compilerDiags = tsLS.getCompilerOptionsDiagnostics();
|
|
|
|
expect(compilerDiags).toEqual([]);
|
|
|
|
const sourceFiles = program.getSourceFiles().filter(f => !f.fileName.endsWith('.d.ts'));
|
|
|
|
// there are six .ts files in the test project
|
|
|
|
expect(sourceFiles.length).toBe(6);
|
|
|
|
for (const {fileName} of sourceFiles) {
|
|
|
|
const syntacticDiags = tsLS.getSyntacticDiagnostics(fileName);
|
|
|
|
expect(syntacticDiags).toEqual([]);
|
|
|
|
const semanticDiags = tsLS.getSemanticDiagnostics(fileName);
|
|
|
|
expect(semanticDiags).toEqual([]);
|
|
|
|
}
|
2016-11-22 12:10:23 -05:00
|
|
|
});
|
|
|
|
|
2019-10-14 18:24:28 -04:00
|
|
|
it('should not report template errors on tour of heroes', () => {
|
|
|
|
const filesWithTemplates = [
|
|
|
|
// Ignore all '*-cases.ts' files as they intentionally contain errors.
|
|
|
|
'/app/app.component.ts',
|
|
|
|
];
|
|
|
|
for (const fileName of filesWithTemplates) {
|
|
|
|
const diags = plugin.getSemanticDiagnostics(fileName);
|
|
|
|
expect(diags).toEqual([]);
|
|
|
|
}
|
2016-11-22 12:10:23 -05:00
|
|
|
});
|
|
|
|
|
2019-10-14 18:24:28 -04:00
|
|
|
it('should respect paths configuration', () => {
|
|
|
|
const SHARED_MODULE = '/app/foo/bar/shared.ts';
|
|
|
|
const MY_COMPONENT = '/app/my.component.ts';
|
|
|
|
mockHost.overrideOptions({
|
|
|
|
baseUrl: '/app',
|
|
|
|
paths: {'bar/*': ['foo/bar/*']},
|
2016-11-22 12:10:23 -05:00
|
|
|
});
|
2019-10-14 18:24:28 -04:00
|
|
|
mockHost.addScript(SHARED_MODULE, `
|
|
|
|
export interface Node {
|
|
|
|
children: Node[];
|
|
|
|
}
|
|
|
|
`);
|
|
|
|
mockHost.addScript(MY_COMPONENT, `
|
|
|
|
import { Component, NgModule } from '@angular/core';
|
|
|
|
import { Node } from 'bar/shared';
|
|
|
|
|
|
|
|
@Component({
|
|
|
|
selector: 'my-component',
|
|
|
|
template: '{{ tree.~{tree}children }}'
|
|
|
|
})
|
|
|
|
export class MyComponent {
|
|
|
|
tree: Node = {
|
|
|
|
children: [],
|
|
|
|
};
|
|
|
|
}
|
2016-11-22 12:10:23 -05:00
|
|
|
|
2019-10-14 18:24:28 -04:00
|
|
|
@NgModule({
|
|
|
|
declarations: [MyComponent],
|
|
|
|
})
|
|
|
|
export class MyModule {}
|
|
|
|
`);
|
|
|
|
// First, make sure there are no errors in newly added scripts.
|
|
|
|
for (const fileName of [SHARED_MODULE, MY_COMPONENT]) {
|
|
|
|
const syntacticDiags = plugin.getSyntacticDiagnostics(fileName);
|
|
|
|
expect(syntacticDiags).toEqual([]);
|
|
|
|
const semanticDiags = plugin.getSemanticDiagnostics(fileName);
|
|
|
|
expect(semanticDiags).toEqual([]);
|
|
|
|
}
|
|
|
|
const marker = mockHost.getLocationMarkerFor(MY_COMPONENT, 'tree');
|
|
|
|
const completions = plugin.getCompletionsAtPosition(MY_COMPONENT, marker.start, undefined);
|
|
|
|
expect(completions).toBeDefined();
|
2020-04-03 23:57:39 -04:00
|
|
|
expect(completions!.entries).toEqual([
|
2019-10-14 18:24:28 -04:00
|
|
|
{
|
|
|
|
name: 'children',
|
|
|
|
kind: CompletionKind.PROPERTY as any,
|
|
|
|
sortText: 'children',
|
2019-10-24 21:29:24 -04:00
|
|
|
replacementSpan: {start: 182, length: 8},
|
2019-11-15 15:06:00 -05:00
|
|
|
insertText: 'children',
|
2019-10-14 18:24:28 -04:00
|
|
|
},
|
|
|
|
]);
|
2016-11-22 12:10:23 -05:00
|
|
|
});
|
2019-10-14 18:24:28 -04:00
|
|
|
});
|
2016-11-22 12:10:23 -05:00
|
|
|
|
2019-10-14 18:24:28 -04:00
|
|
|
describe(`with config 'angularOnly = true`, () => {
|
|
|
|
const mockHost = new MockTypescriptHost(['/app/main.ts']);
|
|
|
|
const tsLS = ts.createLanguageService(mockHost);
|
|
|
|
const plugin = create({
|
|
|
|
languageService: tsLS,
|
|
|
|
languageServiceHost: mockHost,
|
|
|
|
project: mockProject,
|
|
|
|
serverHost: {} as any,
|
|
|
|
config: {
|
|
|
|
angularOnly: true,
|
|
|
|
},
|
2016-11-22 12:10:23 -05:00
|
|
|
});
|
|
|
|
|
2019-10-14 18:24:28 -04:00
|
|
|
it('should not produce TypeScript diagnostics', () => {
|
|
|
|
const fileName = '/foo.ts';
|
|
|
|
mockHost.addScript(fileName, `
|
|
|
|
function add(x: number) {
|
|
|
|
return x + 42;
|
|
|
|
}
|
|
|
|
add('hello');
|
|
|
|
`);
|
|
|
|
const diags = plugin.getSemanticDiagnostics(fileName);
|
|
|
|
expect(diags).toEqual([]);
|
2016-11-22 12:10:23 -05:00
|
|
|
});
|
|
|
|
|
2019-10-14 18:24:28 -04:00
|
|
|
it('should not report template errors on TOH', () => {
|
|
|
|
const filesWithTemplates = [
|
|
|
|
// Ignore all '*-cases.ts' files as they intentionally contain errors.
|
|
|
|
'/app/app.component.ts',
|
|
|
|
'/app/test.ng',
|
|
|
|
];
|
|
|
|
for (const fileName of filesWithTemplates) {
|
|
|
|
const diags = plugin.getSemanticDiagnostics(fileName);
|
|
|
|
expect(diags).toEqual([]);
|
2016-11-22 12:10:23 -05:00
|
|
|
}
|
2019-10-14 18:24:28 -04:00
|
|
|
});
|
|
|
|
});
|