mani-injector
A fast and lightweight dependency injection solution for typescript.
Motivation
The need for a very simple and very fast dependency injection solution to remove as much boilerplate code as possible without sacrificing runtime performance.
Features
- constructor injection only
- class injection
- type injection
- map to instance, singleton, value or provider
- child injector
- small (350 lines of code, 1.27 kb gzipped)
- extendable
Performance
- ~ 40 times faster than InversifyJS
- ~ 5 times faster than TSyringe
- ~ 2.5 times faster than Typed Inject
testet with chrome 75
you can run a basic performance test here (link to project)
Usage
npm install mani-injector --save
Modify your tsconfig.json
to include the following settings
{
"compilerOptions": {
"experimentalDecorators": true,
"emitDecoratorMetadata": true
}
}
Injecting classes
import {Inject, Injector} from 'mani-injector';
class Foo {}
class Bar {}
class FooBar {
constructor(
@Inject readonly foo: Foo,
@Inject readonly bar: Bar
) {}
}
const injector = new Injector();
// new instance is created every time Foo is injected / requested
injector.map(Foo);
// Bar is only created once and that instance is used everywhere
injector.map(Bar).toSingleton();
// new instance is created every time FooBar is injected / requested
injector.map(FooBar);
const fooBar1 = injector.get(FooBar);
const fooBar2 = injector.get(FooBar);
// There are 2 instances of FooBar, 2 instances of Foo and only 1 instance of Bar
console.log(fooBar1.foo !== fooBar2.foo); // true
console.log(fooBar1.bar === fooBar2.bar); // true
Injecting classes with id
import {InjectId, Injector} from 'mani-injector';
class Foo {}
class FooBar {
constructor(
@InjectId('first') readonly foo1: Foo,
@InjectId('second') readonly foo2: Foo,
) {}
}
const injector = new Injector();
// new instance is created every time Foo is requested with the id 'first'
injector.map(Foo, 'first');
// single instance is returned every time Foo is requested with the id 'second'
injector.map(Foo, 'second').toSingleton();
// new instance is created every time FooBar is injected / requested
injector.map(FooBar);
const fooBar1 = injector.get(FooBar);
const fooBar2 = injector.get(FooBar);
// There are 2 instances of FooBar, 2 instances of Foo and only 1 instance of Bar
console.log(fooBar1.foo1 !== fooBar2.foo1); // true
console.log(fooBar1.foo2 === fooBar2.foo2); // true
Injecting types
import {Injector, InjectType} from 'mani-injector';
interface IFoo {}
interface IBar {}
interface IFooBar {
readonly foo: IFoo;
readonly bar: IBar;
}
const barSymbol = Symbol('Bar');
class Foo implements IFoo {}
class Bar implements IBar{}
class FooBar implements IFooBar {
constructor(
@InjectType('foo') readonly foo: IFoo,
@InjectType(barSymbol) readonly bar: IBar
) {}
}
const injector = new Injector();
// every time the type with the id 'foo' is requested a new instance of Foo is created
injector.mapType<IFoo>('foo').toClass(Foo);
// every time the type with the symbol 'barSymbol' is requested a single instance
// of Bar is returned
injector.mapType<IBar>(barSymbol).toSingleton(Bar);
// every time the type with the id 'myFooBar' is requested a new instance of FooBar
// is created
injector.mapType<IFooBar>('myFooBar').toClass(FooBar);
const fooBar1 = injector.getType<IFooBar>('myFooBar');
const fooBar2 = injector.getType<IFooBar>('myFooBar');
// There are 2 instances of FooBar, 2 instances of Foo and only 1 instance of Bar
console.log(fooBar1.foo !== fooBar2.foo); // true
console.log(fooBar1.bar === fooBar2.bar); // true
console.log(fooBar1.bar instanceof Bar); // true
console.log(fooBar1.foo instanceof Foo); // true
Map to value
import {Inject, Injector, InjectType} from 'mani-injector';
class Foo {}
type Bar = { id: number };
class FooBar {
constructor(
@Inject readonly foo: Foo,
@InjectType('bar') readonly bar: Bar,
) {}
}
const injector = new Injector();
const fooValue = new Foo();
const barValue = {id: 1};
// every time Foo is requested return the given value 'fooValue'
injector.map(Foo).toValue(fooValue);
// every time the type with id 'bar' is requested return the given value 'barValue'
injector.mapType<Bar>('bar').toValue(barValue);
// new instance is created every time FooBar is injected / requested
injector.map(FooBar);
const fooBar1 = injector.get(FooBar);
const fooBar2 = injector.get(FooBar);
// There are 2 instances of FooBar, 2 instances of Foo and only 1 instance of Bar
console.log(fooBar1.foo === fooValue); // true
console.log(fooBar2.foo === fooValue); // true
console.log(fooBar1.bar === barValue); // true
console.log(fooBar2.bar === barValue); // true
Map to provider
import {Inject, Injector, InjectType} from 'mani-injector';
interface IBar {
readonly id: number;
}
class Foo {
constructor(readonly id: number) {}
}
class Bar implements IBar {
constructor(readonly id: number) {}
}
class FooBar {
constructor(
@Inject readonly foo: Foo,
@InjectType('bar') readonly bar: IBar,
) {}
}
const injector = new Injector();
let fooCount = 0;
const fooProvider = () => {
return new Foo(fooCount++);
};
let barCount = 0;
const barProvider = () => {
return new Bar(barCount++);
};
// every time Foo is requested the given provider function is called and the
// returned value is injected
injector.map(Foo).toProvider(fooProvider);
// every time the type with id 'bar' is requested the given provider function
// is called and the returned value is injected
injector.mapType<Bar>('bar').toProvider(barProvider);
// new instance is created every time FooBar is injected / requested
injector.map(FooBar);
const fooBar1 = injector.get(FooBar);
const fooBar2 = injector.get(FooBar);
// There are 2 instances of FooBar, 2 instances of Foo and only 1 instance of Bar
console.log(fooBar1.foo.id); // 0
console.log(fooBar2.foo.id); // 1
console.log(fooBar1.bar.id); // 0
console.log(fooBar2.bar.id); // 1
Use child injector
import {Inject, Injector} from 'mani-injector';
class Foo {}
class Bar {}
class FooBar {
constructor(
@Inject readonly foo: Foo,
@Inject readonly bar: Bar
) {}
}
const injector = new Injector();
const childInjector = injector.createChild();
const mainFoo = new Foo();
const childFoo = new Foo();
injector.map(Foo).toValue(mainFoo);
injector.map(Bar).toSingleton();
injector.map(FooBar);
// we override the mapping of Foo to the value childFoo
childInjector.map(Foo).toValue(childFoo);
const mainFooBar = injector.get(FooBar);
// FooBar is requested from the childInjector, the request is forwarded to the
// parent because the child injector has no mapping for it
const childFooBar = childInjector.get(FooBar);
// Bar is mapped in parent injector to a singleton and not overridden by the
// child injector
console.log(mainFooBar.bar === childFooBar.bar);
// Foo is overridden in child injector and mapped to the value childFoo
console.log(childFooBar.foo === childFoo);