2014-10-06 13:45:24 -04:00
|
|
|
import {describe, ddescribe, it, iit, expect, beforeEach} from 'test_lib/test_lib';
|
|
|
|
import {Injector, Inject, InjectLazy, bind} from 'di/di';
|
2014-09-30 14:56:33 -04:00
|
|
|
|
2014-10-07 10:34:07 -04:00
|
|
|
class Engine {
|
|
|
|
}
|
|
|
|
|
2014-10-06 10:13:33 -04:00
|
|
|
class BrokenEngine {
|
|
|
|
constructor() {
|
|
|
|
throw "Broken Engine";
|
|
|
|
}
|
|
|
|
}
|
2014-10-07 10:34:07 -04:00
|
|
|
|
|
|
|
class DashboardSoftware {
|
|
|
|
}
|
|
|
|
|
2014-10-03 17:26:49 -04:00
|
|
|
class Dashboard {
|
2014-10-07 10:34:07 -04:00
|
|
|
constructor(software: DashboardSoftware) {}
|
|
|
|
}
|
|
|
|
|
|
|
|
class TurboEngine extends Engine {
|
2014-10-03 17:26:49 -04:00
|
|
|
}
|
2014-09-30 14:56:33 -04:00
|
|
|
|
|
|
|
class Car {
|
2014-11-22 00:19:23 -05:00
|
|
|
engine:Engine;
|
2014-09-30 14:56:33 -04:00
|
|
|
constructor(engine:Engine) {
|
|
|
|
this.engine = engine;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-10-06 13:45:24 -04:00
|
|
|
class CarWithLazyEngine {
|
2014-11-22 00:19:23 -05:00
|
|
|
engineFactory;
|
2014-10-06 13:45:24 -04:00
|
|
|
constructor(@InjectLazy(Engine) engineFactory) {
|
|
|
|
this.engineFactory = engineFactory;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-30 14:56:33 -04:00
|
|
|
class CarWithDashboard {
|
2014-11-22 00:19:23 -05:00
|
|
|
engine:Engine;
|
|
|
|
dashboard:Dashboard;
|
2014-09-30 14:56:33 -04:00
|
|
|
constructor(engine:Engine, dashboard:Dashboard) {
|
|
|
|
this.engine = engine;
|
|
|
|
this.dashboard = dashboard;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class SportsCar extends Car {
|
2014-11-22 00:19:23 -05:00
|
|
|
engine:Engine;
|
2014-09-30 14:56:33 -04:00
|
|
|
constructor(engine:Engine) {
|
|
|
|
super(engine);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class CarWithInject {
|
2014-11-22 00:19:23 -05:00
|
|
|
engine:Engine;
|
2014-09-30 14:56:33 -04:00
|
|
|
constructor(@Inject(TurboEngine) engine:Engine) {
|
|
|
|
this.engine = engine;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-10-06 10:13:33 -04:00
|
|
|
class CyclicEngine {
|
2014-10-07 10:34:07 -04:00
|
|
|
constructor(car:Car) {}
|
2014-10-06 10:13:33 -04:00
|
|
|
}
|
|
|
|
|
2014-10-05 16:25:42 -04:00
|
|
|
class NoAnnotations {
|
2014-10-07 10:34:07 -04:00
|
|
|
constructor(secretDependency) {}
|
2014-10-05 16:25:42 -04:00
|
|
|
}
|
|
|
|
|
2014-09-30 14:56:33 -04:00
|
|
|
export function main() {
|
2014-10-07 10:34:07 -04:00
|
|
|
describe('injector', function () {
|
|
|
|
it('should instantiate a class without dependencies', function () {
|
2014-09-30 14:56:33 -04:00
|
|
|
var injector = new Injector([Engine]);
|
|
|
|
var engine = injector.get(Engine);
|
|
|
|
|
|
|
|
expect(engine).toBeAnInstanceOf(Engine);
|
|
|
|
});
|
|
|
|
|
2014-10-07 10:34:07 -04:00
|
|
|
it('should resolve dependencies based on type information', function () {
|
2014-09-30 14:56:33 -04:00
|
|
|
var injector = new Injector([Engine, Car]);
|
|
|
|
var car = injector.get(Car);
|
|
|
|
|
|
|
|
expect(car).toBeAnInstanceOf(Car);
|
|
|
|
expect(car.engine).toBeAnInstanceOf(Engine);
|
|
|
|
});
|
|
|
|
|
2014-10-07 10:34:07 -04:00
|
|
|
it('should resolve dependencies based on @Inject annotation', function () {
|
2014-09-30 14:56:33 -04:00
|
|
|
var injector = new Injector([TurboEngine, Engine, CarWithInject]);
|
|
|
|
var car = injector.get(CarWithInject);
|
|
|
|
|
|
|
|
expect(car).toBeAnInstanceOf(CarWithInject);
|
|
|
|
expect(car.engine).toBeAnInstanceOf(TurboEngine);
|
|
|
|
});
|
|
|
|
|
2014-10-05 16:25:42 -04:00
|
|
|
it('should throw when no type and not @Inject', function () {
|
|
|
|
expect(() => new Injector([NoAnnotations])).toThrowError(
|
|
|
|
'Cannot resolve all parameters for NoAnnotations');
|
|
|
|
});
|
|
|
|
|
2014-10-07 10:34:07 -04:00
|
|
|
it('should cache instances', function () {
|
2014-09-30 14:56:33 -04:00
|
|
|
var injector = new Injector([Engine]);
|
|
|
|
|
|
|
|
var e1 = injector.get(Engine);
|
|
|
|
var e2 = injector.get(Engine);
|
|
|
|
|
|
|
|
expect(e1).toBe(e2);
|
|
|
|
});
|
|
|
|
|
2014-10-07 10:34:07 -04:00
|
|
|
it('should bind to a value', function () {
|
2014-09-30 14:56:33 -04:00
|
|
|
var injector = new Injector([
|
|
|
|
bind(Engine).toValue("fake engine")
|
|
|
|
]);
|
|
|
|
|
|
|
|
var engine = injector.get(Engine);
|
|
|
|
expect(engine).toEqual("fake engine");
|
|
|
|
});
|
|
|
|
|
2014-10-07 10:34:07 -04:00
|
|
|
it('should bind to a factory', function () {
|
2014-10-09 11:35:13 -04:00
|
|
|
function sportsCarFactory(e:Engine) {
|
|
|
|
return new SportsCar(e);
|
|
|
|
}
|
|
|
|
|
|
|
|
var injector = new Injector([
|
|
|
|
Engine,
|
|
|
|
bind(Car).toFactory(sportsCarFactory)
|
|
|
|
]);
|
|
|
|
|
|
|
|
var car = injector.get(Car);
|
|
|
|
expect(car).toBeAnInstanceOf(SportsCar);
|
|
|
|
expect(car.engine).toBeAnInstanceOf(Engine);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should support overriding factory dependencies', function () {
|
2014-09-30 14:56:33 -04:00
|
|
|
var injector = new Injector([
|
|
|
|
Engine,
|
2014-10-10 09:56:43 -04:00
|
|
|
bind(Car).toFactory((e) => new SportsCar(e), [Engine])
|
2014-09-30 14:56:33 -04:00
|
|
|
]);
|
|
|
|
|
|
|
|
var car = injector.get(Car);
|
|
|
|
expect(car).toBeAnInstanceOf(SportsCar);
|
|
|
|
expect(car.engine).toBeAnInstanceOf(Engine);
|
|
|
|
});
|
|
|
|
|
2014-10-09 12:18:35 -04:00
|
|
|
it("should flatten passed-in bindings", function () {
|
|
|
|
var injector = new Injector([
|
|
|
|
[[Engine, Car]]
|
|
|
|
]);
|
|
|
|
|
|
|
|
var car = injector.get(Car);
|
|
|
|
expect(car).toBeAnInstanceOf(Car);
|
|
|
|
});
|
|
|
|
|
2015-01-08 12:11:33 -05:00
|
|
|
it("should use the last binding "+
|
|
|
|
"when there are mutliple bindings for same token", function () {
|
|
|
|
var injector = new Injector([
|
|
|
|
bind(Engine).toClass(Engine),
|
|
|
|
bind(Engine).toClass(TurboEngine)
|
|
|
|
]);
|
|
|
|
|
|
|
|
expect(injector.get(Engine)).toBeAnInstanceOf(TurboEngine);
|
|
|
|
});
|
|
|
|
|
2014-10-07 10:34:07 -04:00
|
|
|
it('should use non-type tokens', function () {
|
2014-09-30 14:56:33 -04:00
|
|
|
var injector = new Injector([
|
|
|
|
bind('token').toValue('value')
|
|
|
|
]);
|
|
|
|
|
|
|
|
expect(injector.get('token')).toEqual('value');
|
|
|
|
});
|
|
|
|
|
2014-10-07 10:34:07 -04:00
|
|
|
it('should throw when given invalid bindings', function () {
|
2014-09-30 14:56:33 -04:00
|
|
|
expect(() => new Injector(["blah"])).toThrowError('Invalid binding blah');
|
|
|
|
expect(() => new Injector([bind("blah")])).toThrowError('Invalid binding blah');
|
|
|
|
});
|
|
|
|
|
2014-10-07 10:34:07 -04:00
|
|
|
it('should provide itself', function () {
|
2014-09-30 14:56:33 -04:00
|
|
|
var parent = new Injector([]);
|
|
|
|
var child = parent.createChild([]);
|
|
|
|
|
|
|
|
expect(child.get(Injector)).toBe(child);
|
|
|
|
});
|
|
|
|
|
2014-10-07 10:34:07 -04:00
|
|
|
it('should throw when no provider defined', function () {
|
2014-09-30 14:56:33 -04:00
|
|
|
var injector = new Injector([]);
|
|
|
|
expect(() => injector.get('NonExisting')).toThrowError('No provider for NonExisting!');
|
|
|
|
});
|
|
|
|
|
2014-10-07 10:34:07 -04:00
|
|
|
it('should show the full path when no provider', function () {
|
2014-10-03 17:26:49 -04:00
|
|
|
var injector = new Injector([CarWithDashboard, Engine, Dashboard]);
|
2014-09-30 14:56:33 -04:00
|
|
|
expect(() => injector.get(CarWithDashboard)).
|
2014-10-03 17:26:49 -04:00
|
|
|
toThrowError('No provider for DashboardSoftware! (CarWithDashboard -> Dashboard -> DashboardSoftware)');
|
2014-09-30 14:56:33 -04:00
|
|
|
});
|
2014-10-06 10:13:33 -04:00
|
|
|
|
2014-10-07 10:34:07 -04:00
|
|
|
it('should throw when trying to instantiate a cyclic dependency', function () {
|
2014-10-06 10:13:33 -04:00
|
|
|
var injector = new Injector([
|
|
|
|
Car,
|
|
|
|
bind(Engine).toClass(CyclicEngine)
|
|
|
|
]);
|
|
|
|
|
|
|
|
expect(() => injector.get(Car))
|
|
|
|
.toThrowError('Cannot instantiate cyclic dependency! (Car -> Engine -> Car)');
|
2014-10-06 16:24:12 -04:00
|
|
|
|
|
|
|
expect(() => injector.asyncGet(Car))
|
|
|
|
.toThrowError('Cannot instantiate cyclic dependency! (Car -> Engine -> Car)');
|
2014-10-06 10:13:33 -04:00
|
|
|
});
|
|
|
|
|
2014-10-07 10:34:07 -04:00
|
|
|
it('should show the full path when error happens in a constructor', function () {
|
2014-10-06 10:13:33 -04:00
|
|
|
var injector = new Injector([
|
|
|
|
Car,
|
|
|
|
bind(Engine).toClass(BrokenEngine)
|
|
|
|
]);
|
|
|
|
|
|
|
|
try {
|
|
|
|
injector.get(Car);
|
|
|
|
throw "Must throw";
|
|
|
|
} catch (e) {
|
|
|
|
expect(e.message).toContain("Error during instantiation of Engine! (Car -> Engine)");
|
|
|
|
}
|
|
|
|
});
|
2014-10-06 13:45:24 -04:00
|
|
|
|
2014-10-09 11:03:36 -04:00
|
|
|
it('should instantiate an object after a failed attempt', function () {
|
|
|
|
var isBroken = true;
|
|
|
|
|
|
|
|
var injector = new Injector([
|
|
|
|
Car,
|
2014-10-09 11:35:13 -04:00
|
|
|
bind(Engine).toFactory(() => isBroken ? new BrokenEngine() : new Engine())
|
2014-10-09 11:03:36 -04:00
|
|
|
]);
|
|
|
|
|
2014-10-10 10:00:23 -04:00
|
|
|
expect(() => injector.get(Car)).toThrowError(new RegExp("Error"));
|
2014-10-09 11:03:36 -04:00
|
|
|
|
|
|
|
isBroken = false;
|
|
|
|
|
|
|
|
expect(injector.get(Car)).toBeAnInstanceOf(Car);
|
|
|
|
});
|
|
|
|
|
2014-10-10 11:36:06 -04:00
|
|
|
describe("default bindings", function () {
|
|
|
|
it("should be used when no matching binding found", function () {
|
|
|
|
var injector = new Injector([], {defaultBindings: true});
|
|
|
|
|
|
|
|
var car = injector.get(Car);
|
|
|
|
|
|
|
|
expect(car).toBeAnInstanceOf(Car);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should use the matching binding when it is available", function () {
|
|
|
|
var injector = new Injector([
|
|
|
|
bind(Car).toClass(SportsCar)
|
|
|
|
], {defaultBindings: true});
|
|
|
|
|
|
|
|
var car = injector.get(Car);
|
|
|
|
|
|
|
|
expect(car).toBeAnInstanceOf(SportsCar);
|
|
|
|
});
|
|
|
|
});
|
2014-10-06 13:45:24 -04:00
|
|
|
|
|
|
|
describe("child", function () {
|
2014-10-07 10:34:07 -04:00
|
|
|
it('should load instances from parent injector', function () {
|
2014-10-06 13:45:24 -04:00
|
|
|
var parent = new Injector([Engine]);
|
|
|
|
var child = parent.createChild([]);
|
|
|
|
|
|
|
|
var engineFromParent = parent.get(Engine);
|
|
|
|
var engineFromChild = child.get(Engine);
|
|
|
|
|
|
|
|
expect(engineFromChild).toBe(engineFromParent);
|
|
|
|
});
|
|
|
|
|
2014-10-07 10:34:07 -04:00
|
|
|
it('should create new instance in a child injector', function () {
|
2014-10-06 13:45:24 -04:00
|
|
|
var parent = new Injector([Engine]);
|
|
|
|
var child = parent.createChild([
|
|
|
|
bind(Engine).toClass(TurboEngine)
|
|
|
|
]);
|
|
|
|
|
|
|
|
var engineFromParent = parent.get(Engine);
|
|
|
|
var engineFromChild = child.get(Engine);
|
|
|
|
|
|
|
|
expect(engineFromParent).not.toBe(engineFromChild);
|
|
|
|
expect(engineFromChild).toBeAnInstanceOf(TurboEngine);
|
|
|
|
});
|
2014-10-10 11:36:06 -04:00
|
|
|
|
|
|
|
it("should create child injectors without default bindings", function () {
|
|
|
|
var parent = new Injector([], {defaultBindings: true});
|
|
|
|
var child = parent.createChild([]);
|
|
|
|
|
|
|
|
//child delegates to parent the creation of Car
|
|
|
|
var childCar = child.get(Car);
|
|
|
|
var parentCar = parent.get(Car);
|
|
|
|
|
|
|
|
expect(childCar).toBe(parentCar);
|
|
|
|
});
|
2014-10-06 13:45:24 -04:00
|
|
|
});
|
|
|
|
|
|
|
|
describe("lazy", function () {
|
|
|
|
it("should create dependencies lazily", function () {
|
|
|
|
var injector = new Injector([
|
|
|
|
Engine,
|
|
|
|
CarWithLazyEngine
|
|
|
|
]);
|
|
|
|
|
|
|
|
var car = injector.get(CarWithLazyEngine);
|
|
|
|
expect(car.engineFactory()).toBeAnInstanceOf(Engine);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should cache instance created lazily", function () {
|
|
|
|
var injector = new Injector([
|
|
|
|
Engine,
|
|
|
|
CarWithLazyEngine
|
|
|
|
]);
|
|
|
|
|
|
|
|
var car = injector.get(CarWithLazyEngine);
|
|
|
|
var e1 = car.engineFactory();
|
|
|
|
var e2 = car.engineFactory();
|
|
|
|
|
|
|
|
expect(e1).toBe(e2);
|
|
|
|
});
|
|
|
|
});
|
2014-09-30 14:56:33 -04:00
|
|
|
});
|
2014-10-06 10:13:33 -04:00
|
|
|
}
|