2015-12-15 16:38:27 -08:00
|
|
|
import {Injector, Provider, PLATFORM_INITIALIZER} from 'angular2/core';
|
2015-12-08 19:03:21 -08:00
|
|
|
import {BaseException, ExceptionHandler} from 'angular2/src/facade/exceptions';
|
2015-11-06 17:34:07 -08:00
|
|
|
import {ListWrapper} from 'angular2/src/facade/collection';
|
2015-12-15 16:38:27 -08:00
|
|
|
import {FunctionWrapper, isPresent, Type} from 'angular2/src/facade/lang';
|
2015-12-03 13:31:11 -08:00
|
|
|
|
2015-12-08 19:03:21 -08:00
|
|
|
export class TestInjector {
|
|
|
|
private _instantiated: boolean = false;
|
|
|
|
|
|
|
|
private _injector: Injector = null;
|
|
|
|
|
|
|
|
private _providers: Array<Type | Provider | any[]> = [];
|
|
|
|
|
|
|
|
reset() {
|
|
|
|
this._injector = null;
|
|
|
|
this._providers = [];
|
|
|
|
this._instantiated = false;
|
|
|
|
}
|
|
|
|
|
2015-12-15 16:38:27 -08:00
|
|
|
platformProviders: Array<Type | Provider | any[]> = [];
|
|
|
|
|
|
|
|
applicationProviders: Array<Type | Provider | any[]> = [];
|
|
|
|
|
2015-12-08 19:03:21 -08:00
|
|
|
addProviders(providers: Array<Type | Provider | any[]>) {
|
|
|
|
if (this._instantiated) {
|
|
|
|
throw new BaseException('Cannot add providers after test injector is instantiated');
|
|
|
|
}
|
|
|
|
this._providers = ListWrapper.concat(this._providers, providers);
|
|
|
|
}
|
|
|
|
|
|
|
|
createInjector() {
|
2015-12-15 16:38:27 -08:00
|
|
|
var rootInjector = Injector.resolveAndCreate(this.platformProviders);
|
|
|
|
this._injector = rootInjector.resolveAndCreateChild(
|
|
|
|
ListWrapper.concat(this.applicationProviders, this._providers));
|
2015-12-08 19:03:21 -08:00
|
|
|
this._instantiated = true;
|
|
|
|
return this._injector;
|
|
|
|
}
|
|
|
|
|
|
|
|
execute(fn: FunctionWithParamTokens): any {
|
2015-12-10 12:00:48 -08:00
|
|
|
var additionalProviders = fn.additionalProviders();
|
|
|
|
if (additionalProviders.length > 0) {
|
|
|
|
this.addProviders(additionalProviders);
|
|
|
|
}
|
2015-12-08 19:03:21 -08:00
|
|
|
if (!this._instantiated) {
|
|
|
|
this.createInjector();
|
|
|
|
}
|
|
|
|
return fn.execute(this._injector);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var _testInjector: TestInjector = null;
|
|
|
|
|
|
|
|
export function getTestInjector() {
|
|
|
|
if (_testInjector == null) {
|
|
|
|
_testInjector = new TestInjector();
|
|
|
|
}
|
|
|
|
return _testInjector;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-12-15 16:38:27 -08:00
|
|
|
* Set the providers that the test injector should use. These should be providers
|
|
|
|
* common to every test in the suite.
|
|
|
|
*
|
|
|
|
* This may only be called once, to set up the common providers for the current test
|
|
|
|
* suite on teh current platform. If you absolutely need to change the providers,
|
|
|
|
* first use `resetBaseTestProviders`.
|
|
|
|
*
|
|
|
|
* Test Providers for individual platforms are available from
|
|
|
|
* 'angular2/platform/testing/<platform_name>'.
|
2015-12-08 19:03:21 -08:00
|
|
|
*/
|
2015-12-15 16:38:27 -08:00
|
|
|
export function setBaseTestProviders(platformProviders: Array<Type | Provider | any[]>,
|
|
|
|
applicationProviders: Array<Type | Provider | any[]>) {
|
|
|
|
var testInjector = getTestInjector();
|
|
|
|
if (testInjector.platformProviders.length > 0 || testInjector.applicationProviders.length > 0) {
|
|
|
|
throw new BaseException('Cannot set base providers because it has already been called');
|
|
|
|
}
|
|
|
|
testInjector.platformProviders = platformProviders;
|
|
|
|
testInjector.applicationProviders = applicationProviders;
|
|
|
|
var injector = testInjector.createInjector();
|
|
|
|
let inits: Function[] = injector.getOptional(PLATFORM_INITIALIZER);
|
|
|
|
if (isPresent(inits)) {
|
|
|
|
inits.forEach(init => init());
|
|
|
|
}
|
|
|
|
testInjector.reset();
|
2015-03-13 11:10:11 +01:00
|
|
|
}
|
|
|
|
|
2015-12-08 19:03:21 -08:00
|
|
|
/**
|
2015-12-15 16:38:27 -08:00
|
|
|
* Reset the providers for the test injector.
|
2015-12-08 19:03:21 -08:00
|
|
|
*/
|
2015-12-15 16:38:27 -08:00
|
|
|
export function resetBaseTestProviders() {
|
|
|
|
var testInjector = getTestInjector();
|
|
|
|
testInjector.platformProviders = [];
|
|
|
|
testInjector.applicationProviders = [];
|
|
|
|
testInjector.reset();
|
2015-12-03 13:31:11 -08:00
|
|
|
}
|
|
|
|
|
2015-03-13 20:52:59 +00:00
|
|
|
/**
|
2015-12-08 16:44:04 -08:00
|
|
|
* Allows injecting dependencies in `beforeEach()` and `it()`.
|
2015-03-13 11:10:11 +01:00
|
|
|
*
|
|
|
|
* Example:
|
|
|
|
*
|
2015-03-13 18:48:29 +01:00
|
|
|
* ```
|
|
|
|
* beforeEach(inject([Dependency, AClass], (dep, object) => {
|
|
|
|
* // some code that uses `dep` and `object`
|
|
|
|
* // ...
|
|
|
|
* }));
|
2015-03-13 11:10:11 +01:00
|
|
|
*
|
2015-11-18 18:46:24 -08:00
|
|
|
* it('...', inject([AClass], (object) => {
|
2015-12-08 16:44:04 -08:00
|
|
|
* object.doSomething();
|
|
|
|
* expect(...);
|
2015-03-13 18:48:29 +01:00
|
|
|
* })
|
|
|
|
* ```
|
2015-03-13 11:10:11 +01:00
|
|
|
*
|
|
|
|
* Notes:
|
2015-12-08 19:03:21 -08:00
|
|
|
* - inject is currently a function because of some Traceur limitation the syntax should
|
|
|
|
* eventually
|
2015-03-13 11:10:11 +01:00
|
|
|
* becomes `it('...', @Inject (object: AClass, async: AsyncTestCompleter) => { ... });`
|
|
|
|
*
|
|
|
|
* @param {Array} tokens
|
|
|
|
* @param {Function} fn
|
|
|
|
* @return {FunctionWithParamTokens}
|
|
|
|
*/
|
2015-08-28 11:29:19 -07:00
|
|
|
export function inject(tokens: any[], fn: Function): FunctionWithParamTokens {
|
2015-10-08 15:33:17 -07:00
|
|
|
return new FunctionWithParamTokens(tokens, fn, false);
|
|
|
|
}
|
|
|
|
|
2015-12-10 12:00:48 -08:00
|
|
|
export class InjectSetupWrapper {
|
|
|
|
constructor(private _providers: () => any) {}
|
|
|
|
|
|
|
|
inject(tokens: any[], fn: Function): FunctionWithParamTokens {
|
|
|
|
return new FunctionWithParamTokens(tokens, fn, false, this._providers);
|
|
|
|
}
|
|
|
|
|
|
|
|
injectAsync(tokens: any[], fn: Function): FunctionWithParamTokens {
|
|
|
|
return new FunctionWithParamTokens(tokens, fn, true, this._providers);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export function withProviders(providers: () => any) {
|
|
|
|
return new InjectSetupWrapper(providers);
|
|
|
|
}
|
|
|
|
|
2015-11-18 18:46:24 -08:00
|
|
|
/**
|
2015-12-08 16:44:04 -08:00
|
|
|
* Allows injecting dependencies in `beforeEach()` and `it()`. The test must return
|
|
|
|
* a promise which will resolve when all asynchronous activity is complete.
|
|
|
|
*
|
|
|
|
* Example:
|
|
|
|
*
|
|
|
|
* ```
|
|
|
|
* it('...', injectAsync([AClass], (object) => {
|
|
|
|
* return object.doSomething().then(() => {
|
|
|
|
* expect(...);
|
|
|
|
* });
|
|
|
|
* })
|
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* @param {Array} tokens
|
|
|
|
* @param {Function} fn
|
|
|
|
* @return {FunctionWithParamTokens}
|
2015-11-18 18:46:24 -08:00
|
|
|
*/
|
2015-10-08 15:33:17 -07:00
|
|
|
export function injectAsync(tokens: any[], fn: Function): FunctionWithParamTokens {
|
|
|
|
return new FunctionWithParamTokens(tokens, fn, true);
|
2015-03-13 11:10:11 +01:00
|
|
|
}
|
|
|
|
|
2015-12-10 12:00:48 -08:00
|
|
|
function emptyArray(): Array<any> {
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
2015-03-13 11:10:11 +01:00
|
|
|
export class FunctionWithParamTokens {
|
2015-12-10 12:00:48 -08:00
|
|
|
constructor(private _tokens: any[], private _fn: Function, public isAsync: boolean,
|
|
|
|
public additionalProviders: () => any = emptyArray) {}
|
2015-03-13 11:10:11 +01:00
|
|
|
|
2015-07-17 15:00:52 -07:00
|
|
|
/**
|
|
|
|
* Returns the value of the executed function.
|
|
|
|
*/
|
|
|
|
execute(injector: Injector): any {
|
2015-10-06 18:00:42 -07:00
|
|
|
var params = this._tokens.map(t => injector.get(t));
|
2015-07-17 15:00:52 -07:00
|
|
|
return FunctionWrapper.apply(this._fn, params);
|
2015-03-13 11:10:11 +01:00
|
|
|
}
|
2015-08-28 16:23:28 -07:00
|
|
|
|
|
|
|
hasToken(token: any): boolean { return this._tokens.indexOf(token) > -1; }
|
2015-03-13 11:10:11 +01:00
|
|
|
}
|