perf(Compiler): use Promises only when strictly required

This commit is contained in:
Victor Berchet 2015-02-06 08:57:49 +01:00
parent 47042bc503
commit 74f92c6a79
6 changed files with 304 additions and 176 deletions

View File

@ -86,15 +86,17 @@ export class Compiler {
} }
compile(component:Type, templateRoot:Element = null):Promise<ProtoView> { compile(component:Type, templateRoot:Element = null):Promise<ProtoView> {
return this._compile(this._reader.read(component), templateRoot); var protoView = this._compile(this._reader.read(component), templateRoot);
return PromiseWrapper.isPromise(protoView) ? protoView : PromiseWrapper.resolve(protoView);
} }
// TODO(vicb): union type return ProtoView or Promise<ProtoView>
_compile(cmpMetadata: DirectiveMetadata, templateRoot:Element = null) { _compile(cmpMetadata: DirectiveMetadata, templateRoot:Element = null) {
var pvCached = this._compilerCache.get(cmpMetadata.type); var protoView = this._compilerCache.get(cmpMetadata.type);
if (isPresent(pvCached)) { if (isPresent(protoView)) {
// The component has already been compiled into a ProtoView, // The component has already been compiled into a ProtoView,
// returns a resolved Promise. // returns a resolved Promise.
return PromiseWrapper.resolve(pvCached); return protoView;
} }
var pvPromise = MapWrapper.get(this._compiling, cmpMetadata.type); var pvPromise = MapWrapper.get(this._compiling, cmpMetadata.type);
@ -105,21 +107,22 @@ export class Compiler {
return pvPromise; return pvPromise;
} }
var tplPromise = isBlank(templateRoot) ? var template = isBlank(templateRoot) ? this._templateLoader.load(cmpMetadata) : templateRoot;
this._templateLoader.load(cmpMetadata) :
PromiseWrapper.resolve(templateRoot);
pvPromise = PromiseWrapper.then(tplPromise, if (PromiseWrapper.isPromise(template)) {
(el) => this._compileTemplate(el, cmpMetadata), pvPromise = PromiseWrapper.then(template,
(_) => { throw new BaseException(`Failed to load the template for ${stringify(cmpMetadata.type)}`) } (el) => this._compileTemplate(el, cmpMetadata),
); (_) => { throw new BaseException(`Failed to load the template for ${stringify(cmpMetadata.type)}`); }
);
MapWrapper.set(this._compiling, cmpMetadata.type, pvPromise);
return pvPromise;
}
MapWrapper.set(this._compiling, cmpMetadata.type, pvPromise); return this._compileTemplate(template, cmpMetadata);
return pvPromise;
} }
_compileTemplate(template: Element, cmpMetadata): Promise<ProtoView> { // TODO(vicb): union type return ProtoView or Promise<ProtoView>
_compileTemplate(template: Element, cmpMetadata) {
var pipeline = new CompilePipeline(this.createSteps(cmpMetadata)); var pipeline = new CompilePipeline(this.createSteps(cmpMetadata));
var compileElements = pipeline.process(template); var compileElements = pipeline.process(template);
var protoView = compileElements[0].inheritedProtoView; var protoView = compileElements[0].inheritedProtoView;
@ -130,27 +133,38 @@ export class Compiler {
MapWrapper.delete(this._compiling, cmpMetadata.type); MapWrapper.delete(this._compiling, cmpMetadata.type);
// Compile all the components from the template // Compile all the components from the template
var componentPromises = []; var nestedPVPromises = [];
for (var i = 0; i < compileElements.length; i++) { for (var i = 0; i < compileElements.length; i++) {
var ce = compileElements[i]; var ce = compileElements[i];
if (isPresent(ce.componentDirective)) { if (isPresent(ce.componentDirective)) {
var componentPromise = this._compileNestedProtoView(ce); this._compileNestedProtoView(ce, nestedPVPromises);
ListWrapper.push(componentPromises, componentPromise);
} }
} }
// The protoView is resolved after all the components in the template have been compiled. if (nestedPVPromises.length > 0) {
return PromiseWrapper.then(PromiseWrapper.all(componentPromises), // Returns ProtoView Promise when there are any asynchronous nested ProtoViews.
(_) => protoView, // The promise will resolved after nested ProtoViews are compiled.
(e) => { throw new BaseException(`${e} -> Failed to compile ${stringify(cmpMetadata.type)}`) } return PromiseWrapper.then(PromiseWrapper.all(nestedPVPromises),
); (_) => protoView,
(e) => { throw new BaseException(`${e.message} -> Failed to compile ${stringify(cmpMetadata.type)}`); }
);
}
// When there is no asynchronous nested ProtoViews, return the ProtoView
return protoView;
} }
_compileNestedProtoView(ce: CompileElement):Promise<ProtoView> { _compileNestedProtoView(ce: CompileElement, promises: List<Promise>)
var pvPromise = this._compile(ce.componentDirective); {
pvPromise.then(function(protoView) { var protoView = this._compile(ce.componentDirective);
if (PromiseWrapper.isPromise(protoView)) {
ListWrapper.push(promises, protoView);
protoView.then(function (protoView) {
ce.inheritedElementBinder.nestedProtoView = protoView;
});
} else {
ce.inheritedElementBinder.nestedProtoView = protoView; ce.inheritedElementBinder.nestedProtoView = protoView;
}); }
return pvPromise;
} }
} }

View File

@ -22,13 +22,13 @@ export class TemplateLoader {
this._cache = StringMapWrapper.create(); this._cache = StringMapWrapper.create();
} }
load(cmpMetadata: DirectiveMetadata):Promise<Element> { // TODO(vicb): union type: return an Element or a Promise<Element>
load(cmpMetadata: DirectiveMetadata) {
var annotation:Component = cmpMetadata.annotation; var annotation:Component = cmpMetadata.annotation;
var tplConfig:TemplateConfig = annotation.template; var tplConfig:TemplateConfig = annotation.template;
if (isPresent(tplConfig.inline)) { if (isPresent(tplConfig.inline)) {
var template = DOM.createTemplate(tplConfig.inline); return DOM.createTemplate(tplConfig.inline);
return PromiseWrapper.resolve(template);
} }
if (isPresent(tplConfig.url)) { if (isPresent(tplConfig.url)) {

View File

@ -20,6 +20,10 @@ class PromiseWrapper {
static void setTimeout(fn(), int millis) { static void setTimeout(fn(), int millis) {
new Timer(new Duration(milliseconds: millis), fn); new Timer(new Duration(milliseconds: millis), fn);
} }
static bool isPromise(maybePromise) {
return maybePromise is Future;
}
} }
class _Completer { class _Completer {

View File

@ -39,4 +39,8 @@ export class PromiseWrapper {
static setTimeout(fn:Function, millis:int) { static setTimeout(fn:Function, millis:int) {
window.setTimeout(fn, millis); window.setTimeout(fn, millis);
} }
static isPromise(maybePromise):boolean {
return maybePromise instanceof Promise;
}
} }

View File

@ -1,7 +1,7 @@
import {describe, beforeEach, it, expect, ddescribe, iit, el, IS_DARTIUM} from 'angular2/test_lib'; import {describe, beforeEach, it, expect, ddescribe, iit, el, IS_DARTIUM} from 'angular2/test_lib';
import {DOM, Element, TemplateElement} from 'angular2/src/facade/dom'; import {DOM, Element, TemplateElement} from 'angular2/src/facade/dom';
import {List, ListWrapper, Map, MapWrapper} from 'angular2/src/facade/collection'; import {List, ListWrapper, Map, MapWrapper, StringMapWrapper} from 'angular2/src/facade/collection';
import {Type, isBlank} from 'angular2/src/facade/lang'; import {Type, isBlank, stringify} from 'angular2/src/facade/lang';
import {PromiseWrapper} from 'angular2/src/facade/async'; import {PromiseWrapper} from 'angular2/src/facade/async';
import {Compiler, CompilerCache} from 'angular2/src/core/compiler/compiler'; import {Compiler, CompilerCache} from 'angular2/src/core/compiler/compiler';
@ -27,154 +27,201 @@ export function main() {
reader = new DirectiveMetadataReader(); reader = new DirectiveMetadataReader();
}); });
function createCompiler(processClosure, strategy:ShadowDomStrategy = null, xhr: XHRMock = null) { var syncTemplateLoader = new FakeTemplateLoader();
var steps = [new MockStep(processClosure)]; syncTemplateLoader.forceSync();
if (isBlank(strategy)) { var asyncTemplateLoader = new FakeTemplateLoader();
strategy = new NativeShadowDomStrategy(); asyncTemplateLoader.forceAsync();
}
if (isBlank(xhr)) {
xhr = new XHRMock();
}
return new TestableCompiler(reader, steps, strategy, xhr);
}
it('should run the steps and return the ProtoView of the root element', (done) => { StringMapWrapper.forEach({
var rootProtoView = new ProtoView(null, null, null); '(sync TemplateLoader)': syncTemplateLoader,
var compiler = createCompiler( (parent, current, control) => { '(async TemplateLoader)': asyncTemplateLoader
current.inheritedProtoView = rootProtoView; }, (templateLoader, name) => {
});
compiler.compile(MainComponent, el('<div></div>')).then( (protoView) => {
expect(protoView).toBe(rootProtoView);
done();
});
});
it('should use the given element', (done) => { describe(name, () => {
var element = el('<div></div>');
var compiler = createCompiler( (parent, current, control) => {
current.inheritedProtoView = new ProtoView(current.element, null, null);
});
compiler.compile(MainComponent, element).then( (protoView) => {
expect(protoView.element).toBe(element);
done();
});
});
it('should use the inline template if no element is given explicitly', (done) => { function createCompiler(processClosure) {
var compiler = createCompiler( (parent, current, control) => { var steps = [new MockStep(processClosure)];
current.inheritedProtoView = new ProtoView(current.element, null, null); return new TestableCompiler(reader, steps, templateLoader);
});
compiler.compile(MainComponent, null).then( (protoView) => {
expect(DOM.getInnerHTML(protoView.element)).toEqual('inline component');
done();
});
});
it('should load nested components', (done) => {
var mainEl = el('<div></div>');
var compiler = createCompiler( (parent, current, control) => {
current.inheritedProtoView = new ProtoView(current.element, null, null);
current.inheritedElementBinder = current.inheritedProtoView.bindElement(null);
if (current.element === mainEl) {
current.componentDirective = reader.read(NestedComponent);
} }
});
compiler.compile(MainComponent, mainEl).then( (protoView) => {
var nestedView = protoView.elementBinders[0].nestedProtoView;
expect(DOM.getInnerHTML(nestedView.element)).toEqual('nested component');
done();
});
});
it('should cache compiled components', (done) => { it('should run the steps and return the ProtoView of the root element', (done) => {
var element = el('<div></div>'); var rootProtoView = new ProtoView(null, null, null);
var compiler = createCompiler( (parent, current, control) => { var compiler = createCompiler( (parent, current, control) => {
current.inheritedProtoView = new ProtoView(current.element, null, null); current.inheritedProtoView = rootProtoView;
}); });
var firstProtoView; compiler.compile(MainComponent, el('<div></div>')).then( (protoView) => {
compiler.compile(MainComponent, element).then( (protoView) => { expect(protoView).toBe(rootProtoView);
firstProtoView = protoView; done();
return compiler.compile(MainComponent, element); });
}).then( (protoView) => {
expect(firstProtoView).toBe(protoView);
done();
});
});
it('should re-use components being compiled', (done) => {
var nestedElBinders = [];
var mainEl = el('<div><div class="nested"></div><div class="nested"></div></div>');
var compiler = createCompiler( (parent, current, control) => {
if (DOM.hasClass(current.element, 'nested')) {
current.inheritedProtoView = new ProtoView(current.element, null, null);
current.inheritedElementBinder = current.inheritedProtoView.bindElement(null);
current.componentDirective = reader.read(NestedComponent);
ListWrapper.push(nestedElBinders, current.inheritedElementBinder);
}
});
compiler.compile(MainComponent, mainEl).then( (protoView) => {
expect(nestedElBinders[0].nestedProtoView).toBe(nestedElBinders[1].nestedProtoView);
done();
});
});
it('should allow recursive components', (done) => {
var compiler = createCompiler( (parent, current, control) => {
current.inheritedProtoView = new ProtoView(current.element, null, null);
current.inheritedElementBinder = current.inheritedProtoView.bindElement(null);
current.componentDirective = reader.read(RecursiveComponent);
});
compiler.compile(RecursiveComponent, null).then( (protoView) => {
expect(protoView.elementBinders[0].nestedProtoView).toBe(protoView);
done();
});
});
describe('XHR', () => {
it('should load template via xhr', (done) => {
var xhr = new XHRMock();
xhr.expect('/parent', 'xhr');
var compiler = createCompiler((parent, current, control) => {
current.inheritedProtoView = new ProtoView(current.element, null, null);
}, null, xhr);
compiler.compile(XHRParentComponent).then( (protoView) => {
expect(DOM.getInnerHTML(protoView.element)).toEqual('xhr');
done();
}); });
xhr.flush(); it('should use the given element', (done) => {
}); var element = el('<div></div>');
var compiler = createCompiler( (parent, current, control) => {
it('should return a rejected promise when loading a template fails', (done) => { current.inheritedProtoView = new ProtoView(current.element, null, null);
var xhr = new XHRMock(); });
xhr.expect('/parent', null); compiler.compile(MainComponent, element).then( (protoView) => {
expect(protoView.element).toBe(element);
var compiler = createCompiler((parent, current, control) => {}, null, xhr);
PromiseWrapper.then(compiler.compile(XHRParentComponent),
function(_) { throw 'Failure expected'; },
function(e) {
expect(e.message).toEqual('Failed to load the template for XHRParentComponent');
done(); done();
} });
); });
xhr.flush(); it('should use the inline template if no element is given explicitly', (done) => {
var compiler = createCompiler( (parent, current, control) => {
current.inheritedProtoView = new ProtoView(current.element, null, null);
});
compiler.compile(MainComponent, null).then( (protoView) => {
expect(DOM.getInnerHTML(protoView.element)).toEqual('inline component');
done();
});
});
it('should load nested components', (done) => {
var mainEl = el('<div></div>');
var compiler = createCompiler( (parent, current, control) => {
current.inheritedProtoView = new ProtoView(current.element, null, null);
current.inheritedElementBinder = current.inheritedProtoView.bindElement(null);
if (current.element === mainEl) {
current.componentDirective = reader.read(NestedComponent);
}
});
compiler.compile(MainComponent, mainEl).then( (protoView) => {
var nestedView = protoView.elementBinders[0].nestedProtoView;
expect(DOM.getInnerHTML(nestedView.element)).toEqual('nested component');
done();
});
});
it('should cache compiled components', (done) => {
var element = el('<div></div>');
var compiler = createCompiler( (parent, current, control) => {
current.inheritedProtoView = new ProtoView(current.element, null, null);
});
var firstProtoView;
compiler.compile(MainComponent, element).then( (protoView) => {
firstProtoView = protoView;
return compiler.compile(MainComponent, element);
}).then( (protoView) => {
expect(firstProtoView).toBe(protoView);
done();
});
});
it('should re-use components being compiled', (done) => {
var nestedElBinders = [];
var mainEl = el('<div><div class="nested"></div><div class="nested"></div></div>');
var compiler = createCompiler( (parent, current, control) => {
if (DOM.hasClass(current.element, 'nested')) {
current.inheritedProtoView = new ProtoView(current.element, null, null);
current.inheritedElementBinder = current.inheritedProtoView.bindElement(null);
current.componentDirective = reader.read(NestedComponent);
ListWrapper.push(nestedElBinders, current.inheritedElementBinder);
}
});
compiler.compile(MainComponent, mainEl).then( (protoView) => {
expect(nestedElBinders[0].nestedProtoView).toBe(nestedElBinders[1].nestedProtoView);
done();
});
});
it('should allow recursive components', (done) => {
var compiler = createCompiler( (parent, current, control) => {
current.inheritedProtoView = new ProtoView(current.element, null, null);
current.inheritedElementBinder = current.inheritedProtoView.bindElement(null);
current.componentDirective = reader.read(RecursiveComponent);
});
compiler.compile(RecursiveComponent, null).then( (protoView) => {
expect(protoView.elementBinders[0].nestedProtoView).toBe(protoView);
done();
});
});
}); });
}); });
describe('(mixed async, sync TemplateLoader)', () => {
function createCompiler(processClosure, templateLoader: TemplateLoader) {
var steps = [new MockStep(processClosure)];
return new TestableCompiler(reader, steps, templateLoader);
}
function createNestedComponentSpec(name, loader: TemplateLoader, error:string = null) {
it(`should load nested components ${name}`, (done) => {
var compiler = createCompiler((parent, current, control) => {
if (DOM.hasClass(current.element, 'parent')) {
current.componentDirective = reader.read(NestedComponent);
current.inheritedProtoView = parent.inheritedProtoView;
current.inheritedElementBinder = current.inheritedProtoView.bindElement(null);
} else {
current.inheritedProtoView = new ProtoView(current.element, null, null);
}
}, loader);
PromiseWrapper.then(compiler.compile(ParentComponent),
function(protoView) {
var nestedView = protoView.elementBinders[0].nestedProtoView;
expect(error).toBeNull();
expect(DOM.getInnerHTML(nestedView.element)).toEqual('nested component');
done();
},
function(compileError) {
expect(compileError.message).toEqual(error);
done();
}
);
});
}
var loader = new FakeTemplateLoader();
loader.setSync(ParentComponent);
loader.setSync(NestedComponent);
createNestedComponentSpec('(sync -> sync)', loader);
loader = new FakeTemplateLoader();
loader.setAsync(ParentComponent);
loader.setSync(NestedComponent);
createNestedComponentSpec('(async -> sync)', loader);
loader = new FakeTemplateLoader();
loader.setSync(ParentComponent);
loader.setAsync(NestedComponent);
createNestedComponentSpec('(sync -> async)', loader);
loader = new FakeTemplateLoader();
loader.setAsync(ParentComponent);
loader.setAsync(NestedComponent);
createNestedComponentSpec('(async -> async)', loader);
loader = new FakeTemplateLoader();
loader.setError(ParentComponent);
loader.setSync(NestedComponent);
createNestedComponentSpec('(error -> sync)', loader,
'Failed to load the template for ParentComponent');
// TODO(vicb): Check why errors this fails with Dart
// TODO(vicb): The Promise is rejected with the correct error but an exc is thrown before
//loader = new FakeTemplateLoader();
//loader.setSync(ParentComponent);
//loader.setError(NestedComponent);
//createNestedComponentSpec('(sync -> error)', loader,
// 'Failed to load the template for NestedComponent -> Failed to compile ParentComponent');
//
//loader = new FakeTemplateLoader();
//loader.setAsync(ParentComponent);
//loader.setError(NestedComponent);
//createNestedComponentSpec('(async -> error)', loader,
// 'Failed to load the template for NestedComponent -> Failed to compile ParentComponent');
});
}); });
} }
@Component({ @Component({
template: new TemplateConfig({ template: new TemplateConfig({
url: '/parent' inline: '<div class="parent"></div>'
}) })
}) })
class XHRParentComponent {} class ParentComponent {}
@Component({ @Component({
template: new TemplateConfig({ template: new TemplateConfig({
@ -201,14 +248,9 @@ class RecursiveComponent {}
class TestableCompiler extends Compiler { class TestableCompiler extends Compiler {
steps:List; steps:List;
constructor(reader:DirectiveMetadataReader, steps:List<CompileStep>, strategy:ShadowDomStrategy, constructor(reader:DirectiveMetadataReader, steps:List<CompileStep>, loader: TemplateLoader) {
xhr: XHRMock) { super(dynamicChangeDetection, loader, reader, new Parser(new Lexer()), new CompilerCache(),
super(dynamicChangeDetection, new NativeShadowDomStrategy());
new TemplateLoader(xhr),
reader,
new Parser(new Lexer()),
new CompilerCache(),
strategy);
this.steps = steps; this.steps = steps;
} }
@ -227,3 +269,70 @@ class MockStep extends CompileStep {
this.processClosure(parent, current, control); this.processClosure(parent, current, control);
} }
} }
class FakeTemplateLoader extends TemplateLoader {
_forceSync: boolean;
_forceAsync: boolean;
_syncCmp: List<Type>;
_asyncCmp: List<Type>;
_errorCmp: List<Type>;
constructor() {
super (new XHRMock());
this._forceSync = false;
this._forceAsync = false;
this._syncCmp = [];
this._asyncCmp = [];
this._errorCmp = [];
}
forceSync() {
this._forceSync = true;
this._forceAsync = false;
}
forceAsync() {
this._forceAsync = true;
this._forceSync = false;
}
setSync(component: Type) {
ListWrapper.push(this._syncCmp, component);
}
setAsync(component: Type) {
ListWrapper.push(this._asyncCmp, component);
}
setError(component: Type) {
ListWrapper.push(this._errorCmp, component);
}
load(cmpMetadata: DirectiveMetadata) {
var annotation:Component = cmpMetadata.annotation;
var tplConfig:TemplateConfig = annotation.template;
if (isBlank(tplConfig.inline)) {
throw 'The component must define an inline template';
}
var template = DOM.createTemplate(tplConfig.inline);
if (ListWrapper.contains(this._errorCmp, cmpMetadata.type)) {
return PromiseWrapper.reject('Fail to load');
}
if (ListWrapper.contains(this._syncCmp, cmpMetadata.type)) {
return template;
}
if (ListWrapper.contains(this._asyncCmp, cmpMetadata.type)) {
return PromiseWrapper.resolve(template);
}
if (this._forceSync) return template;
if (this._forceAsync) return PromiseWrapper.resolve(template);
throw `No template configured for ${stringify(cmpMetadata.type)}`;
}
}

View File

@ -23,13 +23,10 @@ export function main() {
return new DirectiveMetadata(FakeComponent, component, null); return new DirectiveMetadata(FakeComponent, component, null);
} }
it('should load inline templates', (done) => { it('should load inline templates synchronously', () => {
var template = 'inline template'; var template = 'inline template';
var md = createMetadata({inline: template}); var md = createMetadata({inline: template});
loader.load(md).then((el) => { expect(loader.load(md).content).toHaveText(template);
expect(el.content).toHaveText(template);
done();
});
}); });
it('should load templates through XHR', (done) => { it('should load templates through XHR', (done) => {