parent
05d66bba3f
commit
ec2d8cc2c8
|
@ -1,4 +1,4 @@
|
||||||
import {Type, isBlank, isPresent, CONST} from 'angular2/src/facade/lang';
|
import {Type, isBlank, isPresent, CONST, BaseException, stringify} from 'angular2/src/facade/lang';
|
||||||
import {List, MapWrapper, ListWrapper} from 'angular2/src/facade/collection';
|
import {List, MapWrapper, ListWrapper} from 'angular2/src/facade/collection';
|
||||||
import {reflector} from 'angular2/src/reflection/reflection';
|
import {reflector} from 'angular2/src/reflection/reflection';
|
||||||
import {Key} from './key';
|
import {Key} from './key';
|
||||||
|
@ -383,7 +383,12 @@ export class BindingBuilder {
|
||||||
* expect(injectorClass.get(Vehicle) instanceof Car).toBe(true);
|
* expect(injectorClass.get(Vehicle) instanceof Car).toBe(true);
|
||||||
* ```
|
* ```
|
||||||
*/
|
*/
|
||||||
toAlias(aliasToken): Binding { return new Binding(this.token, {toAlias: aliasToken}); }
|
toAlias(aliasToken): Binding {
|
||||||
|
if (isBlank(aliasToken)) {
|
||||||
|
throw new BaseException(`Can not alias ${stringify(this.token)} to a blank value!`);
|
||||||
|
}
|
||||||
|
return new Binding(this.token, {toAlias: aliasToken});
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Binds a key to a function which computes the value.
|
* Binds a key to a function which computes the value.
|
||||||
|
|
|
@ -79,16 +79,16 @@ class NoAnnotations {
|
||||||
}
|
}
|
||||||
|
|
||||||
export function main() {
|
export function main() {
|
||||||
describe('injector', function() {
|
describe('injector', () => {
|
||||||
|
|
||||||
it('should instantiate a class without dependencies', function() {
|
it('should instantiate a class without dependencies', () => {
|
||||||
var injector = Injector.resolveAndCreate([Engine]);
|
var injector = Injector.resolveAndCreate([Engine]);
|
||||||
var engine = injector.get(Engine);
|
var engine = injector.get(Engine);
|
||||||
|
|
||||||
expect(engine).toBeAnInstanceOf(Engine);
|
expect(engine).toBeAnInstanceOf(Engine);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should resolve dependencies based on type information', function() {
|
it('should resolve dependencies based on type information', () => {
|
||||||
var injector = Injector.resolveAndCreate([Engine, Car]);
|
var injector = Injector.resolveAndCreate([Engine, Car]);
|
||||||
var car = injector.get(Car);
|
var car = injector.get(Car);
|
||||||
|
|
||||||
|
@ -96,7 +96,7 @@ export function main() {
|
||||||
expect(car.engine).toBeAnInstanceOf(Engine);
|
expect(car.engine).toBeAnInstanceOf(Engine);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should resolve dependencies based on @Inject annotation', function() {
|
it('should resolve dependencies based on @Inject annotation', () => {
|
||||||
var injector = Injector.resolveAndCreate([TurboEngine, Engine, CarWithInject]);
|
var injector = Injector.resolveAndCreate([TurboEngine, Engine, CarWithInject]);
|
||||||
var car = injector.get(CarWithInject);
|
var car = injector.get(CarWithInject);
|
||||||
|
|
||||||
|
@ -104,13 +104,13 @@ export function main() {
|
||||||
expect(car.engine).toBeAnInstanceOf(TurboEngine);
|
expect(car.engine).toBeAnInstanceOf(TurboEngine);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should throw when no type and not @Inject', function() {
|
it('should throw when no type and not @Inject', () => {
|
||||||
expect(() => Injector.resolveAndCreate([NoAnnotations]))
|
expect(() => Injector.resolveAndCreate([NoAnnotations]))
|
||||||
.toThrowError('Cannot resolve all parameters for NoAnnotations. ' +
|
.toThrowError('Cannot resolve all parameters for NoAnnotations. ' +
|
||||||
'Make sure they all have valid type or annotations.');
|
'Make sure they all have valid type or annotations.');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should cache instances', function() {
|
it('should cache instances', () => {
|
||||||
var injector = Injector.resolveAndCreate([Engine]);
|
var injector = Injector.resolveAndCreate([Engine]);
|
||||||
|
|
||||||
var e1 = injector.get(Engine);
|
var e1 = injector.get(Engine);
|
||||||
|
@ -119,14 +119,14 @@ export function main() {
|
||||||
expect(e1).toBe(e2);
|
expect(e1).toBe(e2);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should bind to a value', function() {
|
it('should bind to a value', () => {
|
||||||
var injector = Injector.resolveAndCreate([bind(Engine).toValue("fake engine")]);
|
var injector = Injector.resolveAndCreate([bind(Engine).toValue("fake engine")]);
|
||||||
|
|
||||||
var engine = injector.get(Engine);
|
var engine = injector.get(Engine);
|
||||||
expect(engine).toEqual("fake engine");
|
expect(engine).toEqual("fake engine");
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should bind to a factory', function() {
|
it('should bind to a factory', () => {
|
||||||
function sportsCarFactory(e) { return new SportsCar(e); }
|
function sportsCarFactory(e) { return new SportsCar(e); }
|
||||||
|
|
||||||
var injector =
|
var injector =
|
||||||
|
@ -137,7 +137,7 @@ export function main() {
|
||||||
expect(car.engine).toBeAnInstanceOf(Engine);
|
expect(car.engine).toBeAnInstanceOf(Engine);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should bind to an alias', function() {
|
it('should bind to an alias', () => {
|
||||||
var injector = Injector.resolveAndCreate(
|
var injector = Injector.resolveAndCreate(
|
||||||
[Engine, bind(SportsCar).toClass(SportsCar), bind(Car).toAlias(SportsCar)]);
|
[Engine, bind(SportsCar).toClass(SportsCar), bind(Car).toAlias(SportsCar)]);
|
||||||
|
|
||||||
|
@ -147,13 +147,17 @@ export function main() {
|
||||||
expect(car).toBe(sportsCar);
|
expect(car).toBe(sportsCar);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should throw when the aliased binding does not exist', function() {
|
it('should throw when the aliased binding does not exist', () => {
|
||||||
var injector = Injector.resolveAndCreate([bind('car').toAlias(SportsCar)]);
|
var injector = Injector.resolveAndCreate([bind('car').toAlias(SportsCar)]);
|
||||||
var e = `No provider for ${stringify(SportsCar)}! (car -> ${stringify(SportsCar)})`;
|
var e = `No provider for ${stringify(SportsCar)}! (car -> ${stringify(SportsCar)})`;
|
||||||
expect(() => injector.get('car')).toThrowError(e);
|
expect(() => injector.get('car')).toThrowError(e);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should handle forwardRef in toAlias', function() {
|
it('should throw with a meaningful message when the aliased binding is blank', () => {
|
||||||
|
expect(() => bind('car').toAlias(null)).toThrowError('Can not alias car to a blank value!');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should handle forwardRef in toAlias', () => {
|
||||||
var injector = Injector.resolveAndCreate([
|
var injector = Injector.resolveAndCreate([
|
||||||
bind('originalEngine')
|
bind('originalEngine')
|
||||||
.toClass(forwardRef(() => Engine)),
|
.toClass(forwardRef(() => Engine)),
|
||||||
|
@ -162,7 +166,7 @@ export function main() {
|
||||||
expect(injector.get('aliasedEngine')).toBeAnInstanceOf(Engine);
|
expect(injector.get('aliasedEngine')).toBeAnInstanceOf(Engine);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should support overriding factory dependencies', function() {
|
it('should support overriding factory dependencies', () => {
|
||||||
var injector =
|
var injector =
|
||||||
Injector
|
Injector
|
||||||
.resolveAndCreate([Engine, bind(Car).toFactory((e) => new SportsCar(e), [Engine])]);
|
.resolveAndCreate([Engine, bind(Car).toFactory((e) => new SportsCar(e), [Engine])]);
|
||||||
|
@ -172,35 +176,35 @@ export function main() {
|
||||||
expect(car.engine).toBeAnInstanceOf(Engine);
|
expect(car.engine).toBeAnInstanceOf(Engine);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should support optional dependencies', function() {
|
it('should support optional dependencies', () => {
|
||||||
var injector = Injector.resolveAndCreate([CarWithOptionalEngine]);
|
var injector = Injector.resolveAndCreate([CarWithOptionalEngine]);
|
||||||
|
|
||||||
var car = injector.get(CarWithOptionalEngine);
|
var car = injector.get(CarWithOptionalEngine);
|
||||||
expect(car.engine).toEqual(null);
|
expect(car.engine).toEqual(null);
|
||||||
});
|
});
|
||||||
|
|
||||||
it("should flatten passed-in bindings", function() {
|
it("should flatten passed-in bindings", () => {
|
||||||
var injector = Injector.resolveAndCreate([[[Engine, Car]]]);
|
var injector = Injector.resolveAndCreate([[[Engine, Car]]]);
|
||||||
|
|
||||||
var car = injector.get(Car);
|
var car = injector.get(Car);
|
||||||
expect(car).toBeAnInstanceOf(Car);
|
expect(car).toBeAnInstanceOf(Car);
|
||||||
});
|
});
|
||||||
|
|
||||||
it("should use the last binding " + "when there are multiple bindings for same token",
|
it("should use the last binding when there are multiple bindings for same token", () => {
|
||||||
function() {
|
var injector =
|
||||||
var injector = Injector.resolveAndCreate(
|
Injector
|
||||||
[bind(Engine).toClass(Engine), bind(Engine).toClass(TurboEngine)]);
|
.resolveAndCreate([bind(Engine).toClass(Engine), bind(Engine).toClass(TurboEngine)]);
|
||||||
|
|
||||||
expect(injector.get(Engine)).toBeAnInstanceOf(TurboEngine);
|
expect(injector.get(Engine)).toBeAnInstanceOf(TurboEngine);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should use non-type tokens', function() {
|
it('should use non-type tokens', () => {
|
||||||
var injector = Injector.resolveAndCreate([bind('token').toValue('value')]);
|
var injector = Injector.resolveAndCreate([bind('token').toValue('value')]);
|
||||||
|
|
||||||
expect(injector.get('token')).toEqual('value');
|
expect(injector.get('token')).toEqual('value');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should throw when given invalid bindings', function() {
|
it('should throw when given invalid bindings', () => {
|
||||||
expect(() => Injector.resolveAndCreate(<any>["blah"]))
|
expect(() => Injector.resolveAndCreate(<any>["blah"]))
|
||||||
.toThrowError(
|
.toThrowError(
|
||||||
'Invalid binding - only instances of Binding and Type are allowed, got: blah');
|
'Invalid binding - only instances of Binding and Type are allowed, got: blah');
|
||||||
|
@ -209,26 +213,26 @@ export function main() {
|
||||||
'got: blah');
|
'got: blah');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should provide itself', function() {
|
it('should provide itself', () => {
|
||||||
var parent = Injector.resolveAndCreate([]);
|
var parent = Injector.resolveAndCreate([]);
|
||||||
var child = parent.resolveAndCreateChild([]);
|
var child = parent.resolveAndCreateChild([]);
|
||||||
|
|
||||||
expect(child.get(Injector)).toBe(child);
|
expect(child.get(Injector)).toBe(child);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should throw when no provider defined', function() {
|
it('should throw when no provider defined', () => {
|
||||||
var injector = Injector.resolveAndCreate([]);
|
var injector = Injector.resolveAndCreate([]);
|
||||||
expect(() => injector.get('NonExisting')).toThrowError('No provider for NonExisting!');
|
expect(() => injector.get('NonExisting')).toThrowError('No provider for NonExisting!');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should show the full path when no provider', function() {
|
it('should show the full path when no provider', () => {
|
||||||
var injector = Injector.resolveAndCreate([CarWithDashboard, Engine, Dashboard]);
|
var injector = Injector.resolveAndCreate([CarWithDashboard, Engine, Dashboard]);
|
||||||
expect(() => injector.get(CarWithDashboard))
|
expect(() => injector.get(CarWithDashboard))
|
||||||
.toThrowError(
|
.toThrowError(
|
||||||
`No provider for DashboardSoftware! (${stringify(CarWithDashboard)} -> ${stringify(Dashboard)} -> DashboardSoftware)`);
|
`No provider for DashboardSoftware! (${stringify(CarWithDashboard)} -> ${stringify(Dashboard)} -> DashboardSoftware)`);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should throw when trying to instantiate a cyclic dependency', function() {
|
it('should throw when trying to instantiate a cyclic dependency', () => {
|
||||||
var injector = Injector.resolveAndCreate([Car, bind(Engine).toClass(CyclicEngine)]);
|
var injector = Injector.resolveAndCreate([Car, bind(Engine).toClass(CyclicEngine)]);
|
||||||
|
|
||||||
expect(() => injector.get(Car))
|
expect(() => injector.get(Car))
|
||||||
|
@ -240,7 +244,7 @@ export function main() {
|
||||||
`Cannot instantiate cyclic dependency! (${stringify(Car)} -> ${stringify(Engine)} -> ${stringify(Car)})`);
|
`Cannot instantiate cyclic dependency! (${stringify(Car)} -> ${stringify(Engine)} -> ${stringify(Car)})`);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should show the full path when error happens in a constructor', function() {
|
it('should show the full path when error happens in a constructor', () => {
|
||||||
var injector = Injector.resolveAndCreate([Car, bind(Engine).toClass(BrokenEngine)]);
|
var injector = Injector.resolveAndCreate([Car, bind(Engine).toClass(BrokenEngine)]);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
@ -254,7 +258,7 @@ export function main() {
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should instantiate an object after a failed attempt', function() {
|
it('should instantiate an object after a failed attempt', () => {
|
||||||
var isBroken = true;
|
var isBroken = true;
|
||||||
|
|
||||||
var injector = Injector.resolveAndCreate(
|
var injector = Injector.resolveAndCreate(
|
||||||
|
@ -272,8 +276,8 @@ export function main() {
|
||||||
expect(injector.get('null')).toBe(null);
|
expect(injector.get('null')).toBe(null);
|
||||||
});
|
});
|
||||||
|
|
||||||
describe("default bindings", function() {
|
describe("default bindings", () => {
|
||||||
it("should be used when no matching binding found", function() {
|
it("should be used when no matching binding found", () => {
|
||||||
var injector = Injector.resolveAndCreate([], { defaultBindings: true });
|
var injector = Injector.resolveAndCreate([], { defaultBindings: true });
|
||||||
|
|
||||||
var car = injector.get(Car);
|
var car = injector.get(Car);
|
||||||
|
@ -281,7 +285,7 @@ export function main() {
|
||||||
expect(car).toBeAnInstanceOf(Car);
|
expect(car).toBeAnInstanceOf(Car);
|
||||||
});
|
});
|
||||||
|
|
||||||
it("should use the matching binding when it is available", function() {
|
it("should use the matching binding when it is available", () => {
|
||||||
var injector =
|
var injector =
|
||||||
Injector.resolveAndCreate([bind(Car).toClass(SportsCar)], {defaultBindings: true});
|
Injector.resolveAndCreate([bind(Car).toClass(SportsCar)], {defaultBindings: true});
|
||||||
|
|
||||||
|
@ -291,8 +295,8 @@ export function main() {
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe("child", function() {
|
describe("child", () => {
|
||||||
it('should load instances from parent injector', function() {
|
it('should load instances from parent injector', () => {
|
||||||
var parent = Injector.resolveAndCreate([Engine]);
|
var parent = Injector.resolveAndCreate([Engine]);
|
||||||
var child = parent.resolveAndCreateChild([]);
|
var child = parent.resolveAndCreateChild([]);
|
||||||
|
|
||||||
|
@ -303,7 +307,7 @@ export function main() {
|
||||||
});
|
});
|
||||||
|
|
||||||
it("should not use the child bindings when resolving the dependencies of a parent binding",
|
it("should not use the child bindings when resolving the dependencies of a parent binding",
|
||||||
function() {
|
() => {
|
||||||
var parent = Injector.resolveAndCreate([Car, Engine]);
|
var parent = Injector.resolveAndCreate([Car, Engine]);
|
||||||
var child = parent.resolveAndCreateChild([bind(Engine).toClass(TurboEngine)]);
|
var child = parent.resolveAndCreateChild([bind(Engine).toClass(TurboEngine)]);
|
||||||
|
|
||||||
|
@ -311,7 +315,7 @@ export function main() {
|
||||||
expect(carFromChild.engine).toBeAnInstanceOf(Engine);
|
expect(carFromChild.engine).toBeAnInstanceOf(Engine);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should create new instance in a child injector', function() {
|
it('should create new instance in a child injector', () => {
|
||||||
var parent = Injector.resolveAndCreate([Engine]);
|
var parent = Injector.resolveAndCreate([Engine]);
|
||||||
var child = parent.resolveAndCreateChild([bind(Engine).toClass(TurboEngine)]);
|
var child = parent.resolveAndCreateChild([bind(Engine).toClass(TurboEngine)]);
|
||||||
|
|
||||||
|
@ -322,7 +326,7 @@ export function main() {
|
||||||
expect(engineFromChild).toBeAnInstanceOf(TurboEngine);
|
expect(engineFromChild).toBeAnInstanceOf(TurboEngine);
|
||||||
});
|
});
|
||||||
|
|
||||||
it("should create child injectors without default bindings", function() {
|
it("should create child injectors without default bindings", () => {
|
||||||
var parent = Injector.resolveAndCreate([], { defaultBindings: true });
|
var parent = Injector.resolveAndCreate([], { defaultBindings: true });
|
||||||
var child = parent.resolveAndCreateChild([]);
|
var child = parent.resolveAndCreateChild([]);
|
||||||
|
|
||||||
|
@ -340,15 +344,15 @@ export function main() {
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe("lazy", function() {
|
describe("lazy", () => {
|
||||||
it("should create dependencies lazily", function() {
|
it("should create dependencies lazily", () => {
|
||||||
var injector = Injector.resolveAndCreate([Engine, CarWithLazyEngine]);
|
var injector = Injector.resolveAndCreate([Engine, CarWithLazyEngine]);
|
||||||
|
|
||||||
var car = injector.get(CarWithLazyEngine);
|
var car = injector.get(CarWithLazyEngine);
|
||||||
expect(car.engineFactory()).toBeAnInstanceOf(Engine);
|
expect(car.engineFactory()).toBeAnInstanceOf(Engine);
|
||||||
});
|
});
|
||||||
|
|
||||||
it("should cache instance created lazily", function() {
|
it("should cache instance created lazily", () => {
|
||||||
var injector = Injector.resolveAndCreate([Engine, CarWithLazyEngine]);
|
var injector = Injector.resolveAndCreate([Engine, CarWithLazyEngine]);
|
||||||
|
|
||||||
var car = injector.get(CarWithLazyEngine);
|
var car = injector.get(CarWithLazyEngine);
|
||||||
|
@ -359,7 +363,7 @@ export function main() {
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('resolve', function() {
|
describe('resolve', () => {
|
||||||
it('should resolve and flatten', () => {
|
it('should resolve and flatten', () => {
|
||||||
var bindings = Injector.resolve([Engine, [BrokenEngine]]);
|
var bindings = Injector.resolve([Engine, [BrokenEngine]]);
|
||||||
bindings.forEach(function(b) {
|
bindings.forEach(function(b) {
|
||||||
|
@ -388,7 +392,7 @@ export function main() {
|
||||||
expect(dashboardSoftwareBinding.dependencies[0].key).toEqual(Key.get(BrokenEngine));
|
expect(dashboardSoftwareBinding.dependencies[0].key).toEqual(Key.get(BrokenEngine));
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should support overriding factory dependencies with dependency annotations', function() {
|
it('should support overriding factory dependencies with dependency annotations', () => {
|
||||||
var bindings = Injector.resolve(
|
var bindings = Injector.resolve(
|
||||||
[bind("token")
|
[bind("token")
|
||||||
.toFactory((e) => "result",
|
.toFactory((e) => "result",
|
||||||
|
|
Loading…
Reference in New Issue