With providers split into bundles, the test injector is now able to use providers for a given bundle. Suggested provider lists for tests are available in `angular2/platform/testing/<platform>`. Change the providers for a test suite using `setBaseTestProviders`. This should be done once at the start of the test suite, before any test cases run. BREAKING CHANGE: Tests are now required to use `setBaseTestProviders` to set up. Assuming your tests are run on a browser, setup would change as follows. Before: ```js // Somewhere in test setup import {BrowserDomAdapter} from 'angular2/src/platform/browser/browser_adapter'; BrowserDomAdapter.makeCurrent ``` After: ```js // Somewhere in the test setup import {setBaseTestProviders} from 'angular2/testing'; import { TEST_BROWSER_PLATFORM_PROVIDERS, TEST_BROWSER_APPLICATION_PROVIDERS } from 'angular2/platform/testing/browser'; setBaseTestProviders(TEST_BROWSER_PLATFORM_PROVIDERS, TEST_BROWSER_APPLICATION_PROVIDERS); ``` Closes #5351, Closes #5585 Closes #5975
157 lines
4.6 KiB
TypeScript
157 lines
4.6 KiB
TypeScript
import {Injector, Provider, PLATFORM_INITIALIZER} from 'angular2/core';
|
|
import {BaseException, ExceptionHandler} from 'angular2/src/facade/exceptions';
|
|
import {ListWrapper} from 'angular2/src/facade/collection';
|
|
import {FunctionWrapper, isPresent, Type} from 'angular2/src/facade/lang';
|
|
|
|
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;
|
|
}
|
|
|
|
platformProviders: Array<Type | Provider | any[]> = [];
|
|
|
|
applicationProviders: Array<Type | Provider | any[]> = [];
|
|
|
|
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() {
|
|
var rootInjector = Injector.resolveAndCreate(this.platformProviders);
|
|
this._injector = rootInjector.resolveAndCreateChild(
|
|
ListWrapper.concat(this.applicationProviders, this._providers));
|
|
this._instantiated = true;
|
|
return this._injector;
|
|
}
|
|
|
|
execute(fn: FunctionWithParamTokens): any {
|
|
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;
|
|
}
|
|
|
|
/**
|
|
* 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>'.
|
|
*/
|
|
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();
|
|
}
|
|
|
|
/**
|
|
* Reset the providers for the test injector.
|
|
*/
|
|
export function resetBaseTestProviders() {
|
|
var testInjector = getTestInjector();
|
|
testInjector.platformProviders = [];
|
|
testInjector.applicationProviders = [];
|
|
testInjector.reset();
|
|
}
|
|
|
|
/**
|
|
* Allows injecting dependencies in `beforeEach()` and `it()`.
|
|
*
|
|
* Example:
|
|
*
|
|
* ```
|
|
* beforeEach(inject([Dependency, AClass], (dep, object) => {
|
|
* // some code that uses `dep` and `object`
|
|
* // ...
|
|
* }));
|
|
*
|
|
* it('...', inject([AClass], (object) => {
|
|
* object.doSomething();
|
|
* expect(...);
|
|
* })
|
|
* ```
|
|
*
|
|
* Notes:
|
|
* - inject is currently a function because of some Traceur limitation the syntax should
|
|
* eventually
|
|
* becomes `it('...', @Inject (object: AClass, async: AsyncTestCompleter) => { ... });`
|
|
*
|
|
* @param {Array} tokens
|
|
* @param {Function} fn
|
|
* @return {FunctionWithParamTokens}
|
|
*/
|
|
export function inject(tokens: any[], fn: Function): FunctionWithParamTokens {
|
|
return new FunctionWithParamTokens(tokens, fn, false);
|
|
}
|
|
|
|
/**
|
|
* 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}
|
|
*/
|
|
export function injectAsync(tokens: any[], fn: Function): FunctionWithParamTokens {
|
|
return new FunctionWithParamTokens(tokens, fn, true);
|
|
}
|
|
|
|
export class FunctionWithParamTokens {
|
|
constructor(private _tokens: any[], private _fn: Function, public isAsync: boolean) {}
|
|
|
|
/**
|
|
* Returns the value of the executed function.
|
|
*/
|
|
execute(injector: Injector): any {
|
|
var params = this._tokens.map(t => injector.get(t));
|
|
return FunctionWrapper.apply(this._fn, params);
|
|
}
|
|
|
|
hasToken(token: any): boolean { return this._tokens.indexOf(token) > -1; }
|
|
}
|