fix(core): attempt to recover from user errors during creation (#36381)
If there's an error during the first creation pass of a `TView`, the data structure may be corrupted which will cause framework assertion failures downstream which can mask the user's error. These changes add a new flag to the `TView` that indicates whether the first creation pass was successful, and if it wasn't we try re-create the `TView`. Fixes #31221. PR Close #36381
This commit is contained in:
parent
89c589085d
commit
3d82aa781d
|
@ -30,7 +30,7 @@
|
||||||
"master": {
|
"master": {
|
||||||
"uncompressed": {
|
"uncompressed": {
|
||||||
"runtime-es2015": 1485,
|
"runtime-es2015": 1485,
|
||||||
"main-es2015": 137320,
|
"main-es2015": 137897,
|
||||||
"polyfills-es2015": 37334
|
"polyfills-es2015": 37334
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -49,7 +49,7 @@
|
||||||
"master": {
|
"master": {
|
||||||
"uncompressed": {
|
"uncompressed": {
|
||||||
"runtime-es2015": 2289,
|
"runtime-es2015": 2289,
|
||||||
"main-es2015": 226728,
|
"main-es2015": 227258,
|
||||||
"polyfills-es2015": 36657,
|
"polyfills-es2015": 36657,
|
||||||
"5-es2015": 779
|
"5-es2015": 779
|
||||||
}
|
}
|
||||||
|
|
|
@ -143,6 +143,7 @@ export const TViewConstructor = class TView implements ITView {
|
||||||
public firstChild: ITNode|null, //
|
public firstChild: ITNode|null, //
|
||||||
public schemas: SchemaMetadata[]|null, //
|
public schemas: SchemaMetadata[]|null, //
|
||||||
public consts: TConstants|null, //
|
public consts: TConstants|null, //
|
||||||
|
public incompleteFirstPass: boolean //
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
get template_(): string {
|
get template_(): string {
|
||||||
|
|
|
@ -374,6 +374,14 @@ export function renderView<T>(tView: TView, lView: LView, context: T): void {
|
||||||
renderChildComponents(lView, components);
|
renderChildComponents(lView, components);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
} catch (error) {
|
||||||
|
// If we didn't manage to get past the first template pass due to
|
||||||
|
// an error, mark the view as corrupted so we can try to recover.
|
||||||
|
if (tView.firstCreatePass) {
|
||||||
|
tView.incompleteFirstPass = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
throw error;
|
||||||
} finally {
|
} finally {
|
||||||
lView[FLAGS] &= ~LViewFlags.CreationMode;
|
lView[FLAGS] &= ~LViewFlags.CreationMode;
|
||||||
leaveView();
|
leaveView();
|
||||||
|
@ -598,10 +606,17 @@ export function saveResolvedLocalsInData(
|
||||||
* @returns TView
|
* @returns TView
|
||||||
*/
|
*/
|
||||||
export function getOrCreateTComponentView(def: ComponentDef<any>): TView {
|
export function getOrCreateTComponentView(def: ComponentDef<any>): TView {
|
||||||
return def.tView ||
|
const tView = def.tView;
|
||||||
(def.tView = createTView(
|
|
||||||
TViewType.Component, -1, def.template, def.decls, def.vars, def.directiveDefs,
|
// Create a TView if there isn't one, or recreate it if the first create pass didn't
|
||||||
def.pipeDefs, def.viewQuery, def.schemas, def.consts));
|
// complete successfuly since we can't know for sure whether it's in a usable shape.
|
||||||
|
if (tView === null || tView.incompleteFirstPass) {
|
||||||
|
return def.tView = createTView(
|
||||||
|
TViewType.Component, -1, def.template, def.decls, def.vars, def.directiveDefs,
|
||||||
|
def.pipeDefs, def.viewQuery, def.schemas, def.consts);
|
||||||
|
}
|
||||||
|
|
||||||
|
return tView;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -662,7 +677,9 @@ export function createTView(
|
||||||
typeof pipes === 'function' ? pipes() : pipes, // pipeRegistry: PipeDefList|null,
|
typeof pipes === 'function' ? pipes() : pipes, // pipeRegistry: PipeDefList|null,
|
||||||
null, // firstChild: TNode|null,
|
null, // firstChild: TNode|null,
|
||||||
schemas, // schemas: SchemaMetadata[]|null,
|
schemas, // schemas: SchemaMetadata[]|null,
|
||||||
consts) : // consts: TConstants|null
|
consts, // consts: TConstants|null
|
||||||
|
false // incompleteFirstPass: boolean
|
||||||
|
) :
|
||||||
{
|
{
|
||||||
type: type,
|
type: type,
|
||||||
id: viewIndex,
|
id: viewIndex,
|
||||||
|
@ -694,6 +711,7 @@ export function createTView(
|
||||||
firstChild: null,
|
firstChild: null,
|
||||||
schemas: schemas,
|
schemas: schemas,
|
||||||
consts: consts,
|
consts: consts,
|
||||||
|
incompleteFirstPass: false
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -684,6 +684,12 @@ export interface TView {
|
||||||
* Used for directive matching, attribute bindings, local definitions and more.
|
* Used for directive matching, attribute bindings, local definitions and more.
|
||||||
*/
|
*/
|
||||||
consts: TConstants|null;
|
consts: TConstants|null;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Indicates that there was an error before we managed to complete the first create pass of the
|
||||||
|
* view. This means that the view is likely corrupted and we should try to recover it.
|
||||||
|
*/
|
||||||
|
incompleteFirstPass: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const enum RootContextFlags {
|
export const enum RootContextFlags {
|
||||||
|
|
|
@ -7,7 +7,7 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import {CommonModule} from '@angular/common';
|
import {CommonModule} from '@angular/common';
|
||||||
import {ChangeDetectorRef, Component, ComponentFactoryResolver, Directive, EmbeddedViewRef, Injector, NgModule, TemplateRef, ViewChild, ViewContainerRef, ViewRef} from '@angular/core';
|
import {ChangeDetectorRef, Component, ComponentFactoryResolver, Directive, EmbeddedViewRef, Injector, Input, NgModule, TemplateRef, ViewChild, ViewContainerRef, ViewRef} from '@angular/core';
|
||||||
import {TestBed} from '@angular/core/testing';
|
import {TestBed} from '@angular/core/testing';
|
||||||
import {By} from '@angular/platform-browser';
|
import {By} from '@angular/platform-browser';
|
||||||
import {onlyInIvy} from '@angular/private/testing';
|
import {onlyInIvy} from '@angular/private/testing';
|
||||||
|
@ -281,9 +281,9 @@ describe('view insertion', () => {
|
||||||
|
|
||||||
function createAndInsertViews(beforeTpl: string): any {
|
function createAndInsertViews(beforeTpl: string): any {
|
||||||
TestBed.overrideTemplate(TestCmpt, `
|
TestBed.overrideTemplate(TestCmpt, `
|
||||||
<ng-template #insert>insert</ng-template>
|
<ng-template #insert>insert</ng-template>
|
||||||
<ng-template #before>${beforeTpl}</ng-template>
|
<ng-template #before>${beforeTpl}</ng-template>
|
||||||
|
|
||||||
<div><ng-template #vi="vi" viewInserting></ng-template></div>
|
<div><ng-template #vi="vi" viewInserting></ng-template></div>
|
||||||
`);
|
`);
|
||||||
const fixture = TestBed.createComponent(TestCmpt);
|
const fixture = TestBed.createComponent(TestCmpt);
|
||||||
|
@ -377,12 +377,12 @@ describe('view insertion', () => {
|
||||||
it('should insert before a ng-container with a ViewContainerRef on it', () => {
|
it('should insert before a ng-container with a ViewContainerRef on it', () => {
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'app-root',
|
selector: 'app-root',
|
||||||
template: `
|
template: `
|
||||||
<div>start|</div>
|
<div>start|</div>
|
||||||
<ng-container [ngTemplateOutlet]="insertTpl ? tpl : null"></ng-container>
|
<ng-container [ngTemplateOutlet]="insertTpl ? tpl : null"></ng-container>
|
||||||
<ng-container [ngTemplateOutlet]="tpl"></ng-container>
|
<ng-container [ngTemplateOutlet]="tpl"></ng-container>
|
||||||
<div>|end</div>
|
<div>|end</div>
|
||||||
|
|
||||||
<ng-template #tpl>test</ng-template>
|
<ng-template #tpl>test</ng-template>
|
||||||
`
|
`
|
||||||
})
|
})
|
||||||
|
@ -398,11 +398,11 @@ describe('view insertion', () => {
|
||||||
const fixture = TestBed.createComponent(AppComponent);
|
const fixture = TestBed.createComponent(AppComponent);
|
||||||
fixture.detectChanges();
|
fixture.detectChanges();
|
||||||
|
|
||||||
expect(fixture.debugElement.nativeElement.textContent).toBe('start|test|end');
|
expect(fixture.nativeElement.textContent).toBe('start|test|end');
|
||||||
|
|
||||||
fixture.componentInstance.insertTpl = true;
|
fixture.componentInstance.insertTpl = true;
|
||||||
fixture.detectChanges();
|
fixture.detectChanges();
|
||||||
expect(fixture.debugElement.nativeElement.textContent).toBe('start|testtest|end');
|
expect(fixture.nativeElement.textContent).toBe('start|testtest|end');
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -485,7 +485,7 @@ describe('view insertion', () => {
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'test-cmpt',
|
selector: 'test-cmpt',
|
||||||
template: `
|
template: `
|
||||||
<ng-template #insert>insert</ng-template>
|
<ng-template #insert>insert</ng-template>
|
||||||
<div><ng-template #vi="vi" viewInserting></ng-template></div>
|
<div><ng-template #vi="vi" viewInserting></ng-template></div>
|
||||||
`
|
`
|
||||||
})
|
})
|
||||||
|
@ -614,7 +614,7 @@ describe('view insertion', () => {
|
||||||
it('should properly insert before views in a ViewContainerRef injected on ng-container', () => {
|
it('should properly insert before views in a ViewContainerRef injected on ng-container', () => {
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'app-root',
|
selector: 'app-root',
|
||||||
template: `
|
template: `
|
||||||
<ng-template #parameterListItem let-parameter="parameter">
|
<ng-template #parameterListItem let-parameter="parameter">
|
||||||
{{parameter}}
|
{{parameter}}
|
||||||
</ng-template>
|
</ng-template>
|
||||||
|
@ -643,4 +643,248 @@ describe('view insertion', () => {
|
||||||
expect(fixture.nativeElement.textContent.trim()).toContain('2 1');
|
expect(fixture.nativeElement.textContent.trim()).toContain('2 1');
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('create mode error handling', () => {
|
||||||
|
it('should consistently report errors raised a directive constructor', () => {
|
||||||
|
@Directive({
|
||||||
|
selector: '[failInConstructorAlways]',
|
||||||
|
})
|
||||||
|
class FailInConstructorAlways {
|
||||||
|
constructor() {
|
||||||
|
throw new Error('Error in a constructor');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
template: `<div failInConstructorAlways></div>`,
|
||||||
|
})
|
||||||
|
class TestCmpt {
|
||||||
|
}
|
||||||
|
|
||||||
|
TestBed.configureTestingModule({
|
||||||
|
declarations: [TestCmpt, FailInConstructorAlways],
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(() => {
|
||||||
|
TestBed.createComponent(TestCmpt);
|
||||||
|
}).toThrowError('Error in a constructor');
|
||||||
|
|
||||||
|
expect(() => {
|
||||||
|
TestBed.createComponent(TestCmpt);
|
||||||
|
}).toThrowError('Error in a constructor');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should render even if a directive constructor throws in the first create pass', () => {
|
||||||
|
let firstRun = true;
|
||||||
|
|
||||||
|
@Directive({
|
||||||
|
selector: '[failInConstructorOnce]',
|
||||||
|
})
|
||||||
|
class FailInConstructorOnce {
|
||||||
|
constructor() {
|
||||||
|
if (firstRun) {
|
||||||
|
firstRun = false;
|
||||||
|
throw new Error('Error in a constructor');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
template: `<div failInConstructorOnce>OK</div>`,
|
||||||
|
})
|
||||||
|
class TestCmpt {
|
||||||
|
}
|
||||||
|
|
||||||
|
TestBed.configureTestingModule({
|
||||||
|
declarations: [TestCmpt, FailInConstructorOnce],
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(() => {
|
||||||
|
TestBed.createComponent(TestCmpt);
|
||||||
|
}).toThrowError('Error in a constructor');
|
||||||
|
|
||||||
|
const fixture = TestBed.createComponent(TestCmpt);
|
||||||
|
expect(fixture.nativeElement.textContent).toContain('OK');
|
||||||
|
});
|
||||||
|
|
||||||
|
onlyInIvy('Test depends on static inputs being set during creation')
|
||||||
|
.it('should consistently report errors raised a directive input setter', () => {
|
||||||
|
@Directive({
|
||||||
|
selector: '[failInInputAlways]',
|
||||||
|
})
|
||||||
|
class FailInInputAlways {
|
||||||
|
@Input()
|
||||||
|
set failInInputAlways(_: string) {
|
||||||
|
throw new Error('Error in an input');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
template: `<div failInInputAlways="static"></div>`,
|
||||||
|
})
|
||||||
|
class TestCmpt {
|
||||||
|
}
|
||||||
|
|
||||||
|
TestBed.configureTestingModule({
|
||||||
|
declarations: [TestCmpt, FailInInputAlways],
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(() => {
|
||||||
|
TestBed.createComponent(TestCmpt);
|
||||||
|
}).toThrowError('Error in an input');
|
||||||
|
|
||||||
|
expect(() => {
|
||||||
|
TestBed.createComponent(TestCmpt);
|
||||||
|
}).toThrowError('Error in an input');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should consistently report errors raised a static query setter', () => {
|
||||||
|
@Directive({
|
||||||
|
selector: '[someDir]',
|
||||||
|
})
|
||||||
|
class SomeDirective {
|
||||||
|
}
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
template: `<div someDir></div>`,
|
||||||
|
})
|
||||||
|
class TestCmpt {
|
||||||
|
@ViewChild(SomeDirective, {static: true})
|
||||||
|
set directive(_: SomeDirective) {
|
||||||
|
throw new Error('Error in static query setter');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
TestBed.configureTestingModule({
|
||||||
|
declarations: [TestCmpt, SomeDirective],
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(() => {
|
||||||
|
TestBed.createComponent(TestCmpt);
|
||||||
|
}).toThrowError('Error in static query setter');
|
||||||
|
|
||||||
|
expect(() => {
|
||||||
|
TestBed.createComponent(TestCmpt);
|
||||||
|
}).toThrowError('Error in static query setter');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should match a static query, even if its setter throws in the first create pass', () => {
|
||||||
|
let hasThrown = false;
|
||||||
|
|
||||||
|
@Directive({
|
||||||
|
selector: '[someDir]',
|
||||||
|
})
|
||||||
|
class SomeDirective {
|
||||||
|
}
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
template: `<div someDir></div>`,
|
||||||
|
})
|
||||||
|
class TestCmpt {
|
||||||
|
@ViewChild(SomeDirective, {static: true})
|
||||||
|
get directive() {
|
||||||
|
return this._directive;
|
||||||
|
}
|
||||||
|
set directive(directiveInstance: SomeDirective) {
|
||||||
|
if (!hasThrown) {
|
||||||
|
hasThrown = true;
|
||||||
|
throw new Error('Error in static query setter');
|
||||||
|
}
|
||||||
|
|
||||||
|
this._directive = directiveInstance;
|
||||||
|
}
|
||||||
|
|
||||||
|
private _directive!: SomeDirective;
|
||||||
|
}
|
||||||
|
|
||||||
|
TestBed.configureTestingModule({
|
||||||
|
declarations: [TestCmpt, SomeDirective],
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(() => {
|
||||||
|
TestBed.createComponent(TestCmpt);
|
||||||
|
}).toThrowError('Error in static query setter');
|
||||||
|
|
||||||
|
const fixture = TestBed.createComponent(TestCmpt);
|
||||||
|
|
||||||
|
expect(fixture.componentInstance.directive).toBeInstanceOf(SomeDirective);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should render a recursive component if it throws during the first creation pass', () => {
|
||||||
|
let hasThrown = false;
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
selector: 'test',
|
||||||
|
template: `<ng-content></ng-content>OK`,
|
||||||
|
})
|
||||||
|
class TestCmpt {
|
||||||
|
constructor() {
|
||||||
|
if (!hasThrown) {
|
||||||
|
hasThrown = true;
|
||||||
|
throw new Error('Error in a constructor');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
template: `<test><test><test></test></test></test>`,
|
||||||
|
})
|
||||||
|
class App {
|
||||||
|
}
|
||||||
|
|
||||||
|
TestBed.configureTestingModule({
|
||||||
|
declarations: [App, TestCmpt],
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(() => {
|
||||||
|
TestBed.createComponent(App);
|
||||||
|
}).toThrowError('Error in a constructor');
|
||||||
|
|
||||||
|
const fixture = TestBed.createComponent(App);
|
||||||
|
expect(fixture.nativeElement.textContent).toContain('OKOKOK');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should continue detecting changes if a directive throws in its constructor', () => {
|
||||||
|
let firstRun = true;
|
||||||
|
|
||||||
|
@Directive({
|
||||||
|
selector: '[failInConstructorOnce]',
|
||||||
|
})
|
||||||
|
class FailInConstructorOnce {
|
||||||
|
constructor() {
|
||||||
|
if (firstRun) {
|
||||||
|
firstRun = false;
|
||||||
|
throw new Error('Error in a constructor');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
template: `<div failInConstructorOnce>{{value}}</div>`,
|
||||||
|
})
|
||||||
|
class TestCmpt {
|
||||||
|
value = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
TestBed.configureTestingModule({
|
||||||
|
declarations: [TestCmpt, FailInConstructorOnce],
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(() => {
|
||||||
|
TestBed.createComponent(TestCmpt);
|
||||||
|
}).toThrowError('Error in a constructor');
|
||||||
|
|
||||||
|
const fixture = TestBed.createComponent(TestCmpt);
|
||||||
|
fixture.detectChanges();
|
||||||
|
expect(fixture.nativeElement.textContent).toContain('0');
|
||||||
|
|
||||||
|
fixture.componentInstance.value = 1;
|
||||||
|
fixture.detectChanges();
|
||||||
|
expect(fixture.nativeElement.textContent).toContain('1');
|
||||||
|
|
||||||
|
fixture.componentInstance.value = 2;
|
||||||
|
fixture.detectChanges();
|
||||||
|
expect(fixture.nativeElement.textContent).toContain('2');
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
Loading…
Reference in New Issue