fix(ivy): ensure module scope is rebuild on dependent change (#33522)
During incremental compilations, ngtsc needs to know which metadata from a previous compilation can be reused, versus which metadata has to be recomputed as some dependency was updated. Changes to directives/components should cause the NgModule in which they are declared to be recompiled, as the NgModule's compilation is dependent on its directives/components. When a dependent source file of a directive/component is updated, however, a more subtle dependency should also cause to NgModule's source file to be invalidated. During the reconciliation of state from a previous compilation into the new program, the component's source file is invalidated because one of its dependency has changed, ergo the NgModule needs to be invalidated as well. Up until now, this implicit dependency was not imposed on the NgModule. Additionally, any change to a dependent file may influence the module scope to change, so all components within the module must be invalidated as well. This commit fixes the bug by introducing additional file dependencies, as to ensure a proper rebuild of the module scope and its components. Fixes #32416 PR Close #33522
This commit is contained in:
parent
31c5e67ba6
commit
15f8638b1c
|
@ -69,6 +69,13 @@ export class IncrementalState implements DependencyTracker, MetadataReader, Meta
|
||||||
metadata.fileDependencies.add(dep);
|
metadata.fileDependencies.add(dep);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
trackFileDependencies(deps: ts.SourceFile[], src: ts.SourceFile) {
|
||||||
|
const metadata = this.ensureMetadata(src);
|
||||||
|
for (const dep of deps) {
|
||||||
|
metadata.fileDependencies.add(dep);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
getFileDependencies(file: ts.SourceFile): ts.SourceFile[] {
|
getFileDependencies(file: ts.SourceFile): ts.SourceFile[] {
|
||||||
if (!this.metadata.has(file)) {
|
if (!this.metadata.has(file)) {
|
||||||
return [];
|
return [];
|
||||||
|
|
|
@ -307,13 +307,37 @@ export class IvyCompilation {
|
||||||
const recordSpan = this.perf.start('recordDependencies');
|
const recordSpan = this.perf.start('recordDependencies');
|
||||||
this.scopeRegistry !.getCompilationScopes().forEach(scope => {
|
this.scopeRegistry !.getCompilationScopes().forEach(scope => {
|
||||||
const file = scope.declaration.getSourceFile();
|
const file = scope.declaration.getSourceFile();
|
||||||
// Register the file containing the NgModule where the declaration is declared.
|
const ngModuleFile = scope.ngModule.getSourceFile();
|
||||||
this.incrementalState.trackFileDependency(scope.ngModule.getSourceFile(), file);
|
|
||||||
scope.directives.forEach(
|
// A change to any dependency of the declaration causes the declaration to be invalidated,
|
||||||
directive =>
|
// which requires the NgModule to be invalidated as well.
|
||||||
this.incrementalState.trackFileDependency(directive.ref.node.getSourceFile(), file));
|
const deps = this.incrementalState.getFileDependencies(file);
|
||||||
scope.pipes.forEach(
|
this.incrementalState.trackFileDependencies(deps, ngModuleFile);
|
||||||
pipe => this.incrementalState.trackFileDependency(pipe.ref.node.getSourceFile(), file));
|
|
||||||
|
// A change to the NgModule file should cause the declaration itself to be invalidated.
|
||||||
|
this.incrementalState.trackFileDependency(ngModuleFile, file);
|
||||||
|
|
||||||
|
// A change to any directive/pipe in the compilation scope should cause the declaration to be
|
||||||
|
// invalidated.
|
||||||
|
scope.directives.forEach(directive => {
|
||||||
|
const dirSf = directive.ref.node.getSourceFile();
|
||||||
|
|
||||||
|
// When a directive in scope is updated, the declaration needs to be recompiled as e.g.
|
||||||
|
// a selector may have changed.
|
||||||
|
this.incrementalState.trackFileDependency(dirSf, file);
|
||||||
|
|
||||||
|
// When any of the dependencies of the declaration changes, the NgModule scope may be
|
||||||
|
// affected so a component within scope must be recompiled. Only components need to be
|
||||||
|
// recompiled, as directives are not dependent upon the compilation scope.
|
||||||
|
if (directive.isComponent) {
|
||||||
|
this.incrementalState.trackFileDependencies(deps, dirSf);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
scope.pipes.forEach(pipe => {
|
||||||
|
// When a pipe in scope is updated, the declaration needs to be recompiled as e.g.
|
||||||
|
// the pipe's name may have changed.
|
||||||
|
this.incrementalState.trackFileDependency(pipe.ref.node.getSourceFile(), file);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
this.perf.stop(recordSpan);
|
this.perf.stop(recordSpan);
|
||||||
}
|
}
|
||||||
|
|
|
@ -165,6 +165,62 @@ runInEachFileSystem(() => {
|
||||||
expect(written).not.toContain('/foo_module.js');
|
expect(written).not.toContain('/foo_module.js');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// https://github.com/angular/angular/issues/32416
|
||||||
|
it('should rebuild full NgModule scope when a dependency of a declaration has changed', () => {
|
||||||
|
env.write('component1.ts', `
|
||||||
|
import {Component} from '@angular/core';
|
||||||
|
import {SELECTOR} from './dep';
|
||||||
|
|
||||||
|
@Component({selector: SELECTOR, template: 'cmp'})
|
||||||
|
export class Cmp1 {}
|
||||||
|
`);
|
||||||
|
env.write('component2.ts', `
|
||||||
|
import {Component} from '@angular/core';
|
||||||
|
|
||||||
|
@Component({selector: 'cmp2', template: 'cmp2'})
|
||||||
|
export class Cmp2 {}
|
||||||
|
`);
|
||||||
|
env.write('dep.ts', `
|
||||||
|
export const SELECTOR = 'cmp';
|
||||||
|
`);
|
||||||
|
env.write('directive.ts', `
|
||||||
|
import {Directive} from '@angular/core';
|
||||||
|
|
||||||
|
@Directive({selector: 'dir'})
|
||||||
|
export class Dir {}
|
||||||
|
`);
|
||||||
|
env.write('pipe.ts', `
|
||||||
|
import {Pipe} from '@angular/core';
|
||||||
|
|
||||||
|
@Pipe({name: 'myPipe'})
|
||||||
|
export class MyPipe {}
|
||||||
|
`);
|
||||||
|
env.write('module.ts', `
|
||||||
|
import {NgModule} from '@angular/core';
|
||||||
|
import {Cmp1} from './component1';
|
||||||
|
import {Cmp2} from './component2';
|
||||||
|
import {Dir} from './directive';
|
||||||
|
import {MyPipe} from './pipe';
|
||||||
|
|
||||||
|
@NgModule({declarations: [Cmp1, Cmp2, Dir, MyPipe]})
|
||||||
|
export class Mod {}
|
||||||
|
`);
|
||||||
|
env.driveMain();
|
||||||
|
|
||||||
|
// Pretend a change was made to 'dep'. Since this may affect the NgModule scope, like it does
|
||||||
|
// here if the selector is updated, all components in the module scope need to be recompiled.
|
||||||
|
env.flushWrittenFileTracking();
|
||||||
|
env.invalidateCachedFile('dep.ts');
|
||||||
|
env.driveMain();
|
||||||
|
const written = env.getFilesWrittenSinceLastFlush();
|
||||||
|
expect(written).not.toContain('/directive.js');
|
||||||
|
expect(written).not.toContain('/pipe.js');
|
||||||
|
expect(written).toContain('/component1.js');
|
||||||
|
expect(written).toContain('/component2.js');
|
||||||
|
expect(written).toContain('/dep.js');
|
||||||
|
expect(written).toContain('/module.js');
|
||||||
|
});
|
||||||
|
|
||||||
it('should rebuild components where their NgModule declared dependencies have changed', () => {
|
it('should rebuild components where their NgModule declared dependencies have changed', () => {
|
||||||
setupFooBarProgram(env);
|
setupFooBarProgram(env);
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue