This library implements dependency injection for javascript.
It is currently work in progress and in unstable beta phase
but the API should not change anymore before 1.0.0 stable release
will arrived.
The Container API
Creating a container
The container is the place where all dependencies get bound to. We can have
multiple containers in our project in parallel.
import {Container} from "@owja/ioc";
const container = new Container();
Binding
Binding a class
This is the default way to bind a dependency. The class will get instantiated when the
dependency gets resolved.
container.bind<ServiceInterface>(symbol).to(Service);
Binding a class in singleton scope
This will create only one instance of Service
container.bind<ServiceInterface>(symbol).to(Service).inSingletonScope();
Binding a factory
Factories are functions which will get called when the dependency gets resolved
container.bind<ServiceInterface>(symbol).toFactory(() => new Service());
container.bind<string>(symbol).toFactory(() => "just a string");
A factory can configured for singleton scope too. This way will only executed once.
container.bind<ServiceInterface>(symbol).toFactory(() => new Service()).inSingletonScope();
Binding a value
This is always like singleton scope, but it should be avoid to instantiate
dependencies here. If they are circular dependencies, they will fail.
container.bind<ServiceInterface>(symbol).toValue(new Service());
container.bind<string>(symbol).toValue("just a string");
container.bind<() => string>(symbol).toValue(() => "i am a function");
Rebinding
This is the way how we can rebind a dependency while unit tests. We should not need to
rebind in production code.
container.rebind<ServiceMock>(symbol).toValue(new ServiceMock());
Removing
Normally this function is not used in production code. This will remove the
dependency from the container.
container.remove(symbol);
Getting a dependency
Getting dependencies without inject
decorators are only meant for unit tests. This is also
the internal way the inject
decorator gets the dependency it has to resolve.
container.get<Interface>(symbol);
Snapshot & Restore
This creates a snapshot of the bound dependencies. After this we can rebind dependencies
and can restore it back to its old state after we made some unit tests.
container.snapshot();
container.restore();
The inject
Decorator
We have to create a inject
decorator for each container.
import {createDecorator} from "@owja/ioc";
export const inject = createDecorator(container);
This decorator is needed to resolve our dependencies.
class Example {
@inject(symbol)
readonly service!: Interface;
}
The symbol
Symbols are used to identify our dependencies. A good practice is to keep them in one place.
export const TYPE = {
"Service" = Symbol.for("Service"),
}
Usage
Step 1 - Installing the OWJA! IoC library
npm install --save-dev @owja/ioc
Step 2 - Create symbols for our dependencies
Now we create the folder services and add the new file services/types.ts:
export const TYPE = {
"MyService" = Symbol.for("MyService"),
"MyOtherService" = Symbol.for("MyOtherService"),
};
Step 3 - Creating a container
Next we need a container to bind our dependencies to. Let's create the file services/container.ts
import {Container, createDecorator} from "@owja/ioc";
import {TYPE} from "./types";
import {IMyService, MyService} from "./service/my-service";
import {IMyOtherService, MyOtherService} from "./service/my-other-service";
const container = new Container();
const inject = createDecorator(container);
container.bind<IMyService>(TYPE.MyService).to(MyService);
container.bind<IMyOtherService>(TYPE.MyOtherService).to(MyOtherService);
export {container, TYPE, inject};
Step 4 - Injecting dependencies
Lets create a example.ts file in our source root:
import {container, TYPE, inject} from "./services/container";
import {IMyService} from "./service/my-service";
import {IMyOtherService} from "./service/my-other-service";
class Example {
@inject(TYPE.MyService)
readonly myService!: IMyService;
@inject(TYPE.MyOtherSerice)
readonly myOtherService!: IMyOtherService;
}
const example = new Example();
console.log(example.myService);
console.log(example.myOtherSerice);
If we run this example we should see the content of our example services.
The dependencies (services) will injected on the first call. This means if you rebind the service after
accessing the properties of the Example class, it will not resolve the new service. If you want a new
service each time you call example.myService
you have to add the NOCACHE
tag:
import {container, TYPE, inject} from "./services/container";
import {NOCACHE} from "@owja/ioc";
class Example {
@inject(TYPE.MyService, NOCACHE)
readonly myService!: IMyService;
@inject(TYPE.MyOtherSerice, NOCACHE)
readonly myOtherService!: IMyOtherService;
}
Unit testing with IoC
We can snapshot and restore a container for unit testing.
We are able to make multiple snapshots in a row too.
import {container, TYPE} from "./services/container";
beforeEach(() => {
container.snapshot();
});
afterEach(() => {
container.restore();
}
test("can do something", () => {
container.rebind<MyServiceMock>(TYPE.MySerice).to(MyServiceMock);
const mock = container.get<MyServiceMock>(TYPE.MySerice);
});
Development
We are working on the first stable release. Current state of development can be seen in our
Github Project for the first release.
Inspiration
This library is highly inspired by InversifyJS
but has other goals:
- Make the library very lightweight (less than one kilobyte)
- Implementing less features to make the API more straight forward
- Always lazy inject the dependencies
- No meta-reflect required
License
License under Creative Commons Attribution 4.0 International
Copyright © 2019 Hauke Broer