feat(binding): throw on binding to a blank alias

fixes #2068
This commit is contained in:
Victor Berchet 2015-05-26 10:55:12 +02:00
parent 05d66bba3f
commit ec2d8cc2c8
2 changed files with 52 additions and 43 deletions

View File

@ -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.

View File

@ -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",