perf(language-service): update NgCompiler via resource-only path when able (#40585)
This commit changes the Language Service's "compiler factory" mechanism to leverage the new resource-only update path for `NgCompiler`. When an incoming change only affects a resource file like a component template or stylesheet, going through the new API allows the Language Service to avoid unnecessary incremental steps of the `NgCompiler` and return answers more efficiently. PR Close #40585
This commit is contained in:
parent
11ca2f04f9
commit
e3bd23c915
|
@ -25,7 +25,13 @@ export class LanguageServiceAdapter implements NgCompilerAdapter {
|
||||||
readonly factoryTracker = null; // no .ngfactory shims
|
readonly factoryTracker = null; // no .ngfactory shims
|
||||||
readonly unifiedModulesHost = null; // only used in Bazel
|
readonly unifiedModulesHost = null; // only used in Bazel
|
||||||
readonly rootDirs: AbsoluteFsPath[];
|
readonly rootDirs: AbsoluteFsPath[];
|
||||||
private readonly templateVersion = new Map<string, string>();
|
|
||||||
|
/**
|
||||||
|
* Map of resource filenames to the version of the file last read via `readResource`.
|
||||||
|
*
|
||||||
|
* Used to implement `getModifiedResourceFiles`.
|
||||||
|
*/
|
||||||
|
private readonly lastReadResourceVersion = new Map<string, string>();
|
||||||
|
|
||||||
constructor(private readonly project: ts.server.Project) {
|
constructor(private readonly project: ts.server.Project) {
|
||||||
this.rootDirs = getRootDirs(this, project.getCompilationSettings());
|
this.rootDirs = getRootDirs(this, project.getCompilationSettings());
|
||||||
|
@ -81,14 +87,18 @@ export class LanguageServiceAdapter implements NgCompilerAdapter {
|
||||||
throw new Error(`Failed to get script snapshot while trying to read ${fileName}`);
|
throw new Error(`Failed to get script snapshot while trying to read ${fileName}`);
|
||||||
}
|
}
|
||||||
const version = this.project.getScriptVersion(fileName);
|
const version = this.project.getScriptVersion(fileName);
|
||||||
this.templateVersion.set(fileName, version);
|
this.lastReadResourceVersion.set(fileName, version);
|
||||||
return snapshot.getText(0, snapshot.getLength());
|
return snapshot.getText(0, snapshot.getLength());
|
||||||
}
|
}
|
||||||
|
|
||||||
isTemplateDirty(fileName: string): boolean {
|
getModifiedResourceFiles(): Set<string>|undefined {
|
||||||
const lastVersion = this.templateVersion.get(fileName);
|
const modifiedFiles = new Set<string>();
|
||||||
const latestVersion = this.project.getScriptVersion(fileName);
|
for (const [fileName, oldVersion] of this.lastReadResourceVersion) {
|
||||||
return lastVersion !== latestVersion;
|
if (this.project.getScriptVersion(fileName) !== oldVersion) {
|
||||||
|
modifiedFiles.add(fileName);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return modifiedFiles.size > 0 ? modifiedFiles : undefined;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -6,7 +6,7 @@
|
||||||
* found in the LICENSE file at https://angular.io/license
|
* found in the LICENSE file at https://angular.io/license
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import {CompilationTicket, freshCompilationTicket, incrementalFromCompilerTicket, NgCompiler} from '@angular/compiler-cli/src/ngtsc/core';
|
import {CompilationTicket, freshCompilationTicket, incrementalFromCompilerTicket, NgCompiler, resourceChangeTicket} from '@angular/compiler-cli/src/ngtsc/core';
|
||||||
import {NgCompilerOptions} from '@angular/compiler-cli/src/ngtsc/core/api';
|
import {NgCompilerOptions} from '@angular/compiler-cli/src/ngtsc/core/api';
|
||||||
import {TrackedIncrementalBuildStrategy} from '@angular/compiler-cli/src/ngtsc/incremental';
|
import {TrackedIncrementalBuildStrategy} from '@angular/compiler-cli/src/ngtsc/incremental';
|
||||||
import {TypeCheckingProgramStrategy} from '@angular/compiler-cli/src/ngtsc/typecheck/api';
|
import {TypeCheckingProgramStrategy} from '@angular/compiler-cli/src/ngtsc/typecheck/api';
|
||||||
|
@ -37,53 +37,32 @@ export class CompilerFactory {
|
||||||
|
|
||||||
getOrCreate(): NgCompiler {
|
getOrCreate(): NgCompiler {
|
||||||
const program = this.programStrategy.getProgram();
|
const program = this.programStrategy.getProgram();
|
||||||
if (this.compiler === null || program !== this.lastKnownProgram) {
|
const modifiedResourceFiles = this.adapter.getModifiedResourceFiles() ?? new Set();
|
||||||
let ticket: CompilationTicket;
|
|
||||||
if (this.compiler === null || this.lastKnownProgram === null) {
|
if (this.compiler !== null && program === this.lastKnownProgram) {
|
||||||
ticket = freshCompilationTicket(
|
if (modifiedResourceFiles.size > 0) {
|
||||||
program, this.options, this.incrementalStrategy, this.programStrategy, true, true);
|
// Only resource files have changed since the last NgCompiler was created.
|
||||||
} else {
|
const ticket = resourceChangeTicket(this.compiler, modifiedResourceFiles);
|
||||||
ticket = incrementalFromCompilerTicket(
|
this.compiler = NgCompiler.fromTicket(ticket, this.adapter);
|
||||||
this.compiler, program, this.incrementalStrategy, this.programStrategy, new Set());
|
|
||||||
}
|
}
|
||||||
this.compiler = NgCompiler.fromTicket(ticket, this.adapter);
|
|
||||||
this.lastKnownProgram = program;
|
return this.compiler;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let ticket: CompilationTicket;
|
||||||
|
if (this.compiler === null || this.lastKnownProgram === null) {
|
||||||
|
ticket = freshCompilationTicket(
|
||||||
|
program, this.options, this.incrementalStrategy, this.programStrategy, true, true);
|
||||||
|
} else {
|
||||||
|
ticket = incrementalFromCompilerTicket(
|
||||||
|
this.compiler, program, this.incrementalStrategy, this.programStrategy,
|
||||||
|
modifiedResourceFiles);
|
||||||
|
}
|
||||||
|
this.compiler = NgCompiler.fromTicket(ticket, this.adapter);
|
||||||
|
this.lastKnownProgram = program;
|
||||||
return this.compiler;
|
return this.compiler;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Create a new instance of the Ivy compiler if the program has changed since
|
|
||||||
* the last time the compiler was instantiated. If the program has not changed,
|
|
||||||
* return the existing instance.
|
|
||||||
* @param fileName override the template if this is an external template file
|
|
||||||
* @param options angular compiler options
|
|
||||||
*/
|
|
||||||
getOrCreateWithChangedFile(fileName: string): NgCompiler {
|
|
||||||
const compiler = this.getOrCreate();
|
|
||||||
if (isExternalTemplate(fileName)) {
|
|
||||||
this.overrideTemplate(fileName, compiler);
|
|
||||||
}
|
|
||||||
return compiler;
|
|
||||||
}
|
|
||||||
|
|
||||||
private overrideTemplate(fileName: string, compiler: NgCompiler) {
|
|
||||||
if (!this.adapter.isTemplateDirty(fileName)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
// 1. Get the latest snapshot
|
|
||||||
const latestTemplate = this.adapter.readResource(fileName);
|
|
||||||
// 2. Find all components that use the template
|
|
||||||
const ttc = compiler.getTemplateTypeChecker();
|
|
||||||
const components = compiler.getComponentsWithTemplateFile(fileName);
|
|
||||||
// 3. Update component template
|
|
||||||
for (const component of components) {
|
|
||||||
if (ts.isClassDeclaration(component)) {
|
|
||||||
ttc.overrideComponentTemplate(component, latestTemplate);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
registerLastKnownProgram() {
|
registerLastKnownProgram() {
|
||||||
this.lastKnownProgram = this.programStrategy.getProgram();
|
this.lastKnownProgram = this.programStrategy.getProgram();
|
||||||
}
|
}
|
||||||
|
|
|
@ -67,7 +67,7 @@ export class LanguageService {
|
||||||
}
|
}
|
||||||
|
|
||||||
getSemanticDiagnostics(fileName: string): ts.Diagnostic[] {
|
getSemanticDiagnostics(fileName: string): ts.Diagnostic[] {
|
||||||
const compiler = this.compilerFactory.getOrCreateWithChangedFile(fileName);
|
const compiler = this.compilerFactory.getOrCreate();
|
||||||
const ttc = compiler.getTemplateTypeChecker();
|
const ttc = compiler.getTemplateTypeChecker();
|
||||||
const diagnostics: ts.Diagnostic[] = [];
|
const diagnostics: ts.Diagnostic[] = [];
|
||||||
if (isTypeScriptFile(fileName)) {
|
if (isTypeScriptFile(fileName)) {
|
||||||
|
@ -90,7 +90,7 @@ export class LanguageService {
|
||||||
|
|
||||||
getDefinitionAndBoundSpan(fileName: string, position: number): ts.DefinitionInfoAndBoundSpan
|
getDefinitionAndBoundSpan(fileName: string, position: number): ts.DefinitionInfoAndBoundSpan
|
||||||
|undefined {
|
|undefined {
|
||||||
const compiler = this.compilerFactory.getOrCreateWithChangedFile(fileName);
|
const compiler = this.compilerFactory.getOrCreate();
|
||||||
const results =
|
const results =
|
||||||
new DefinitionBuilder(this.tsLS, compiler).getDefinitionAndBoundSpan(fileName, position);
|
new DefinitionBuilder(this.tsLS, compiler).getDefinitionAndBoundSpan(fileName, position);
|
||||||
this.compilerFactory.registerLastKnownProgram();
|
this.compilerFactory.registerLastKnownProgram();
|
||||||
|
@ -99,7 +99,7 @@ export class LanguageService {
|
||||||
|
|
||||||
getTypeDefinitionAtPosition(fileName: string, position: number):
|
getTypeDefinitionAtPosition(fileName: string, position: number):
|
||||||
readonly ts.DefinitionInfo[]|undefined {
|
readonly ts.DefinitionInfo[]|undefined {
|
||||||
const compiler = this.compilerFactory.getOrCreateWithChangedFile(fileName);
|
const compiler = this.compilerFactory.getOrCreate();
|
||||||
const results =
|
const results =
|
||||||
new DefinitionBuilder(this.tsLS, compiler).getTypeDefinitionsAtPosition(fileName, position);
|
new DefinitionBuilder(this.tsLS, compiler).getTypeDefinitionsAtPosition(fileName, position);
|
||||||
this.compilerFactory.registerLastKnownProgram();
|
this.compilerFactory.registerLastKnownProgram();
|
||||||
|
@ -107,7 +107,7 @@ export class LanguageService {
|
||||||
}
|
}
|
||||||
|
|
||||||
getQuickInfoAtPosition(fileName: string, position: number): ts.QuickInfo|undefined {
|
getQuickInfoAtPosition(fileName: string, position: number): ts.QuickInfo|undefined {
|
||||||
const compiler = this.compilerFactory.getOrCreateWithChangedFile(fileName);
|
const compiler = this.compilerFactory.getOrCreate();
|
||||||
const templateInfo = getTemplateInfoAtPosition(fileName, position, compiler);
|
const templateInfo = getTemplateInfoAtPosition(fileName, position, compiler);
|
||||||
if (templateInfo === undefined) {
|
if (templateInfo === undefined) {
|
||||||
return undefined;
|
return undefined;
|
||||||
|
@ -129,7 +129,7 @@ export class LanguageService {
|
||||||
}
|
}
|
||||||
|
|
||||||
getReferencesAtPosition(fileName: string, position: number): ts.ReferenceEntry[]|undefined {
|
getReferencesAtPosition(fileName: string, position: number): ts.ReferenceEntry[]|undefined {
|
||||||
const compiler = this.compilerFactory.getOrCreateWithChangedFile(fileName);
|
const compiler = this.compilerFactory.getOrCreate();
|
||||||
const results = new ReferencesAndRenameBuilder(this.strategy, this.tsLS, compiler)
|
const results = new ReferencesAndRenameBuilder(this.strategy, this.tsLS, compiler)
|
||||||
.getReferencesAtPosition(fileName, position);
|
.getReferencesAtPosition(fileName, position);
|
||||||
this.compilerFactory.registerLastKnownProgram();
|
this.compilerFactory.registerLastKnownProgram();
|
||||||
|
@ -137,7 +137,7 @@ export class LanguageService {
|
||||||
}
|
}
|
||||||
|
|
||||||
getRenameInfo(fileName: string, position: number): ts.RenameInfo {
|
getRenameInfo(fileName: string, position: number): ts.RenameInfo {
|
||||||
const compiler = this.compilerFactory.getOrCreateWithChangedFile(fileName);
|
const compiler = this.compilerFactory.getOrCreate();
|
||||||
const renameInfo = new ReferencesAndRenameBuilder(this.strategy, this.tsLS, compiler)
|
const renameInfo = new ReferencesAndRenameBuilder(this.strategy, this.tsLS, compiler)
|
||||||
.getRenameInfo(absoluteFrom(fileName), position);
|
.getRenameInfo(absoluteFrom(fileName), position);
|
||||||
if (!renameInfo.canRename) {
|
if (!renameInfo.canRename) {
|
||||||
|
@ -152,7 +152,7 @@ export class LanguageService {
|
||||||
}
|
}
|
||||||
|
|
||||||
findRenameLocations(fileName: string, position: number): readonly ts.RenameLocation[]|undefined {
|
findRenameLocations(fileName: string, position: number): readonly ts.RenameLocation[]|undefined {
|
||||||
const compiler = this.compilerFactory.getOrCreateWithChangedFile(fileName);
|
const compiler = this.compilerFactory.getOrCreate();
|
||||||
const results = new ReferencesAndRenameBuilder(this.strategy, this.tsLS, compiler)
|
const results = new ReferencesAndRenameBuilder(this.strategy, this.tsLS, compiler)
|
||||||
.findRenameLocations(fileName, position);
|
.findRenameLocations(fileName, position);
|
||||||
this.compilerFactory.registerLastKnownProgram();
|
this.compilerFactory.registerLastKnownProgram();
|
||||||
|
@ -161,7 +161,7 @@ export class LanguageService {
|
||||||
|
|
||||||
private getCompletionBuilder(fileName: string, position: number):
|
private getCompletionBuilder(fileName: string, position: number):
|
||||||
CompletionBuilder<TmplAstNode|AST>|null {
|
CompletionBuilder<TmplAstNode|AST>|null {
|
||||||
const compiler = this.compilerFactory.getOrCreateWithChangedFile(fileName);
|
const compiler = this.compilerFactory.getOrCreate();
|
||||||
const templateInfo = getTemplateInfoAtPosition(fileName, position, compiler);
|
const templateInfo = getTemplateInfoAtPosition(fileName, position, compiler);
|
||||||
if (templateInfo === undefined) {
|
if (templateInfo === undefined) {
|
||||||
return null;
|
return null;
|
||||||
|
@ -219,7 +219,7 @@ export class LanguageService {
|
||||||
}
|
}
|
||||||
|
|
||||||
getTcb(fileName: string, position: number): GetTcbResponse {
|
getTcb(fileName: string, position: number): GetTcbResponse {
|
||||||
return this.withCompiler<GetTcbResponse>(fileName, compiler => {
|
return this.withCompiler<GetTcbResponse>(compiler => {
|
||||||
const templateInfo = getTemplateInfoAtPosition(fileName, position, compiler);
|
const templateInfo = getTemplateInfoAtPosition(fileName, position, compiler);
|
||||||
if (templateInfo === undefined) {
|
if (templateInfo === undefined) {
|
||||||
return undefined;
|
return undefined;
|
||||||
|
@ -263,8 +263,8 @@ export class LanguageService {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
private withCompiler<T>(fileName: string, p: (compiler: NgCompiler) => T): T {
|
private withCompiler<T>(p: (compiler: NgCompiler) => T): T {
|
||||||
const compiler = this.compilerFactory.getOrCreateWithChangedFile(fileName);
|
const compiler = this.compilerFactory.getOrCreate();
|
||||||
const result = p(compiler);
|
const result = p(compiler);
|
||||||
this.compilerFactory.registerLastKnownProgram();
|
this.compilerFactory.registerLastKnownProgram();
|
||||||
return result;
|
return result;
|
||||||
|
|
|
@ -17,6 +17,38 @@ describe('language-service/compiler integration', () => {
|
||||||
initMockFileSystem('Native');
|
initMockFileSystem('Native');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('should react to a change in an external template', () => {
|
||||||
|
const cmpFile = absoluteFrom('/test.ts');
|
||||||
|
const tmplFile = absoluteFrom('/test.html');
|
||||||
|
|
||||||
|
const env = LanguageServiceTestEnvironment.setup([
|
||||||
|
{
|
||||||
|
name: cmpFile,
|
||||||
|
contents: `
|
||||||
|
import {Component} from '@angular/core';
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
selector: 'test-cmp',
|
||||||
|
templateUrl: './test.html',
|
||||||
|
})
|
||||||
|
export class TestCmp {}
|
||||||
|
`,
|
||||||
|
isRoot: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: tmplFile,
|
||||||
|
contents: '<other-cmp>Test</other-cmp>',
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
|
||||||
|
const diags = env.ngLS.getSemanticDiagnostics(cmpFile);
|
||||||
|
expect(diags.length).toBeGreaterThan(0);
|
||||||
|
|
||||||
|
env.updateFile(tmplFile, '<div>Test</div>');
|
||||||
|
const afterDiags = env.ngLS.getSemanticDiagnostics(cmpFile);
|
||||||
|
expect(afterDiags.length).toBe(0);
|
||||||
|
});
|
||||||
|
|
||||||
it('should not produce errors from inline test declarations mixing with those of the app', () => {
|
it('should not produce errors from inline test declarations mixing with those of the app', () => {
|
||||||
const appCmpFile = absoluteFrom('/test.cmp.ts');
|
const appCmpFile = absoluteFrom('/test.cmp.ts');
|
||||||
const appModuleFile = absoluteFrom('/test.mod.ts');
|
const appModuleFile = absoluteFrom('/test.mod.ts');
|
||||||
|
|
|
@ -6,14 +6,13 @@
|
||||||
* found in the LICENSE file at https://angular.io/license
|
* found in the LICENSE file at https://angular.io/license
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import {TmplAstNode} from '@angular/compiler';
|
|
||||||
import {absoluteFrom, AbsoluteFsPath} from '@angular/compiler-cli/src/ngtsc/file_system';
|
import {absoluteFrom, AbsoluteFsPath} from '@angular/compiler-cli/src/ngtsc/file_system';
|
||||||
import {initMockFileSystem} from '@angular/compiler-cli/src/ngtsc/file_system/testing';
|
import {initMockFileSystem} from '@angular/compiler-cli/src/ngtsc/file_system/testing';
|
||||||
import * as ts from 'typescript';
|
import * as ts from 'typescript';
|
||||||
import {DisplayInfoKind, unsafeCastDisplayInfoKindToScriptElementKind} from '../display_parts';
|
import {DisplayInfoKind, unsafeCastDisplayInfoKindToScriptElementKind} from '../display_parts';
|
||||||
import {LanguageService} from '../language_service';
|
import {LanguageService} from '../language_service';
|
||||||
|
|
||||||
import {LanguageServiceTestEnvironment} from './env';
|
import {extractCursorInfo, LanguageServiceTestEnvironment} from './env';
|
||||||
|
|
||||||
const DIR_WITH_INPUT = {
|
const DIR_WITH_INPUT = {
|
||||||
'Dir': `
|
'Dir': `
|
||||||
|
@ -640,7 +639,6 @@ function setup(
|
||||||
AppCmp: ts.ClassDeclaration,
|
AppCmp: ts.ClassDeclaration,
|
||||||
ngLS: LanguageService,
|
ngLS: LanguageService,
|
||||||
cursor: number,
|
cursor: number,
|
||||||
nodes: TmplAstNode[],
|
|
||||||
text: string,
|
text: string,
|
||||||
} {
|
} {
|
||||||
const codePath = absoluteFrom('/test.ts');
|
const codePath = absoluteFrom('/test.ts');
|
||||||
|
@ -650,6 +648,7 @@ function setup(
|
||||||
|
|
||||||
const otherDirectiveClassDecls = Object.values(otherDeclarations).join('\n\n');
|
const otherDirectiveClassDecls = Object.values(otherDeclarations).join('\n\n');
|
||||||
|
|
||||||
|
const {cursor, text: templateWithoutCursor} = extractCursorInfo(templateWithCursor);
|
||||||
const env = LanguageServiceTestEnvironment.setup([
|
const env = LanguageServiceTestEnvironment.setup([
|
||||||
{
|
{
|
||||||
name: codePath,
|
name: codePath,
|
||||||
|
@ -675,18 +674,15 @@ function setup(
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: templatePath,
|
name: templatePath,
|
||||||
contents: 'Placeholder template',
|
contents: templateWithoutCursor,
|
||||||
}
|
}
|
||||||
]);
|
]);
|
||||||
const {nodes, cursor, text} =
|
|
||||||
env.overrideTemplateWithCursor(codePath, 'AppCmp', templateWithCursor);
|
|
||||||
return {
|
return {
|
||||||
env,
|
env,
|
||||||
fileName: templatePath,
|
fileName: templatePath,
|
||||||
AppCmp: env.getClass(codePath, 'AppCmp'),
|
AppCmp: env.getClass(codePath, 'AppCmp'),
|
||||||
ngLS: env.ngLS,
|
ngLS: env.ngLS,
|
||||||
nodes,
|
text: templateWithoutCursor,
|
||||||
text,
|
|
||||||
cursor,
|
cursor,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
|
@ -27,8 +27,8 @@ describe('definitions', () => {
|
||||||
};
|
};
|
||||||
|
|
||||||
const env = createModuleWithDeclarations([appFile], [templateFile]);
|
const env = createModuleWithDeclarations([appFile], [templateFile]);
|
||||||
const {cursor} = env.overrideTemplateWithCursor(
|
const {cursor} = env.updateFileWithCursor(
|
||||||
absoluteFrom('/app.ts'), 'AppCmp', '<input #myInput /> {{myIn¦put.value}}');
|
absoluteFrom('/app.html'), '<input #myInput /> {{myIn¦put.value}}');
|
||||||
env.expectNoSourceDiagnostics();
|
env.expectNoSourceDiagnostics();
|
||||||
const {definitions} = env.ngLS.getDefinitionAndBoundSpan(absoluteFrom('/app.html'), cursor)!;
|
const {definitions} = env.ngLS.getDefinitionAndBoundSpan(absoluteFrom('/app.html'), cursor)!;
|
||||||
expect(definitions![0].name).toEqual('myInput');
|
expect(definitions![0].name).toEqual('myInput');
|
||||||
|
|
|
@ -46,13 +46,6 @@ function writeTsconfig(
|
||||||
|
|
||||||
export type TestableOptions = StrictTemplateOptions;
|
export type TestableOptions = StrictTemplateOptions;
|
||||||
|
|
||||||
export interface TemplateOverwriteResult {
|
|
||||||
cursor: number;
|
|
||||||
nodes: TmplAstNode[];
|
|
||||||
component: ts.ClassDeclaration;
|
|
||||||
text: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
export class LanguageServiceTestEnvironment {
|
export class LanguageServiceTestEnvironment {
|
||||||
private constructor(
|
private constructor(
|
||||||
readonly tsLS: ts.LanguageService, readonly ngLS: LanguageService,
|
readonly tsLS: ts.LanguageService, readonly ngLS: LanguageService,
|
||||||
|
@ -118,26 +111,16 @@ export class LanguageServiceTestEnvironment {
|
||||||
return getClassOrError(sf, className);
|
return getClassOrError(sf, className);
|
||||||
}
|
}
|
||||||
|
|
||||||
overrideTemplateWithCursor(fileName: AbsoluteFsPath, className: string, contents: string):
|
updateFileWithCursor(fileName: AbsoluteFsPath, contents: string): {cursor: number, text: string} {
|
||||||
TemplateOverwriteResult {
|
|
||||||
const program = this.tsLS.getProgram();
|
|
||||||
if (program === undefined) {
|
|
||||||
throw new Error(`Expected to get a ts.Program`);
|
|
||||||
}
|
|
||||||
const sf = getSourceFileOrError(program, fileName);
|
|
||||||
const component = getClassOrError(sf, className);
|
|
||||||
|
|
||||||
const ngCompiler = this.ngLS.compilerFactory.getOrCreate();
|
|
||||||
const templateTypeChecker = ngCompiler.getTemplateTypeChecker();
|
|
||||||
|
|
||||||
const {cursor, text} = extractCursorInfo(contents);
|
const {cursor, text} = extractCursorInfo(contents);
|
||||||
|
this.updateFile(fileName, text);
|
||||||
const {nodes} = templateTypeChecker.overrideComponentTemplate(component, text);
|
return {cursor, text};
|
||||||
return {cursor, nodes, component, text};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
updateFile(fileName: AbsoluteFsPath, contents: string): void {
|
updateFile(fileName: AbsoluteFsPath, contents: string): void {
|
||||||
const scriptInfo = this.projectService.getScriptInfo(fileName);
|
const normalFileName = ts.server.toNormalizedPath(fileName);
|
||||||
|
const scriptInfo =
|
||||||
|
this.projectService.getOrCreateScriptInfoForNormalizedPath(normalFileName, true, '');
|
||||||
if (scriptInfo === undefined) {
|
if (scriptInfo === undefined) {
|
||||||
throw new Error(`Could not find a file named ${fileName}`);
|
throw new Error(`Could not find a file named ${fileName}`);
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,41 +0,0 @@
|
||||||
/**
|
|
||||||
* @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
|
|
||||||
*/
|
|
||||||
|
|
||||||
import * as ts from 'typescript/lib/tsserverlibrary';
|
|
||||||
|
|
||||||
import {LanguageServiceAdapter} from '../../adapters';
|
|
||||||
|
|
||||||
import {MockService, setup, TEST_TEMPLATE} from './mock_host';
|
|
||||||
|
|
||||||
describe('Language service adapter', () => {
|
|
||||||
let project: ts.server.Project;
|
|
||||||
let service: MockService;
|
|
||||||
|
|
||||||
beforeAll(() => {
|
|
||||||
const {project: _project, service: _service} = setup();
|
|
||||||
project = _project;
|
|
||||||
service = _service;
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should mark template dirty if it has not seen the template before', () => {
|
|
||||||
const adapter = new LanguageServiceAdapter(project);
|
|
||||||
expect(adapter.isTemplateDirty(TEST_TEMPLATE)).toBeTrue();
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should not mark template dirty if template has not changed', () => {
|
|
||||||
const adapter = new LanguageServiceAdapter(project);
|
|
||||||
adapter.readResource(TEST_TEMPLATE);
|
|
||||||
expect(adapter.isTemplateDirty(TEST_TEMPLATE)).toBeFalse();
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should mark template dirty if template has changed', () => {
|
|
||||||
const adapter = new LanguageServiceAdapter(project);
|
|
||||||
service.overwrite(TEST_TEMPLATE, '<p>Hello World</p>');
|
|
||||||
expect(adapter.isTemplateDirty(TEST_TEMPLATE)).toBeTrue();
|
|
||||||
});
|
|
||||||
});
|
|
|
@ -6,7 +6,7 @@
|
||||||
* found in the LICENSE file at https://angular.io/license
|
* found in the LICENSE file at https://angular.io/license
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import {absoluteFrom} from '@angular/compiler-cli/src/ngtsc/file_system';
|
import {absoluteFrom, AbsoluteFsPath} from '@angular/compiler-cli/src/ngtsc/file_system';
|
||||||
import {initMockFileSystem, TestFile} from '@angular/compiler-cli/src/ngtsc/file_system/testing';
|
import {initMockFileSystem, TestFile} from '@angular/compiler-cli/src/ngtsc/file_system/testing';
|
||||||
|
|
||||||
import * as ts from 'typescript/lib/tsserverlibrary';
|
import * as ts from 'typescript/lib/tsserverlibrary';
|
||||||
|
@ -476,8 +476,8 @@ describe('quick info', () => {
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should provide documentation', () => {
|
it('should provide documentation', () => {
|
||||||
const {cursor} = env.overrideTemplateWithCursor(
|
const {cursor} =
|
||||||
absoluteFrom('/app.ts'), 'AppCmp', `<div>{{¦title}}</div>`);
|
env.updateFileWithCursor(absoluteFrom('/app.html'), `<div>{{¦title}}</div>`);
|
||||||
const quickInfo = env.ngLS.getQuickInfoAtPosition(absoluteFrom('/app.html'), cursor);
|
const quickInfo = env.ngLS.getQuickInfoAtPosition(absoluteFrom('/app.html'), cursor);
|
||||||
const documentation = toText(quickInfo!.documentation);
|
const documentation = toText(quickInfo!.documentation);
|
||||||
expect(documentation).toBe('This is the title of the `AppCmp` Component.');
|
expect(documentation).toBe('This is the title of the `AppCmp` Component.');
|
||||||
|
@ -522,8 +522,7 @@ describe('quick info', () => {
|
||||||
{templateOverride, expectedSpanText, expectedDisplayString}:
|
{templateOverride, expectedSpanText, expectedDisplayString}:
|
||||||
{templateOverride: string, expectedSpanText: string, expectedDisplayString: string}):
|
{templateOverride: string, expectedSpanText: string, expectedDisplayString: string}):
|
||||||
ts.QuickInfo {
|
ts.QuickInfo {
|
||||||
const {cursor, text} =
|
const {cursor, text} = env.updateFileWithCursor(absoluteFrom('/app.html'), templateOverride);
|
||||||
env.overrideTemplateWithCursor(absoluteFrom('/app.ts'), 'AppCmp', templateOverride);
|
|
||||||
env.expectNoSourceDiagnostics();
|
env.expectNoSourceDiagnostics();
|
||||||
env.expectNoTemplateDiagnostics(absoluteFrom('/app.ts'), 'AppCmp');
|
env.expectNoTemplateDiagnostics(absoluteFrom('/app.ts'), 'AppCmp');
|
||||||
const quickInfo = env.ngLS.getQuickInfoAtPosition(absoluteFrom('/app.html'), cursor);
|
const quickInfo = env.ngLS.getQuickInfoAtPosition(absoluteFrom('/app.html'), cursor);
|
||||||
|
|
|
@ -49,8 +49,7 @@ describe('type definitions', () => {
|
||||||
});
|
});
|
||||||
|
|
||||||
function getTypeDefinitionsAndAssertBoundSpan({templateOverride}: {templateOverride: string}) {
|
function getTypeDefinitionsAndAssertBoundSpan({templateOverride}: {templateOverride: string}) {
|
||||||
const {cursor, text} =
|
const {cursor, text} = env.updateFileWithCursor(absoluteFrom('/app.html'), templateOverride);
|
||||||
env.overrideTemplateWithCursor(absoluteFrom('/app.ts'), 'AppCmp', templateOverride);
|
|
||||||
env.expectNoSourceDiagnostics();
|
env.expectNoSourceDiagnostics();
|
||||||
env.expectNoTemplateDiagnostics(absoluteFrom('/app.ts'), 'AppCmp');
|
env.expectNoTemplateDiagnostics(absoluteFrom('/app.ts'), 'AppCmp');
|
||||||
const defs = env.ngLS.getTypeDefinitionAtPosition(absoluteFrom('/app.html'), cursor);
|
const defs = env.ngLS.getTypeDefinitionAtPosition(absoluteFrom('/app.html'), cursor);
|
||||||
|
|
Loading…
Reference in New Issue