Tiny Injector
Tiny Injector is a tiny yet powerful and flexible Dependency Injection library
Tiny Injector
Tiny Injector is a tiny yet powerful and flexible Dependency Injection library for projects that uses TypeScript. It could be used on top of existing projects althought it is recommended to be used as first-class citizen
The work heavily inspired by . NET Dependency Injection, Angular +2 and This Answer.
Parts of documentation are taken from the Microsoft DI website
What Is Dependency Injection
In software engineering, dependency injection is a technique in which an object receives other objects that it depends on. These other objects are called dependencies.
In the typical "using" relationship the receiving object is called a client and the passed (that is, "injected") object is called a service.
Reference You might check here And here as well
in which an object receives other objects that it depends on.
Let's say you have two classes ObjectA
and ObjectB
, and ObjectB
depends on ObjectA
, in this case we call ObjectA
dependency on ObjectB
.
In the typical "using" relationship the receiving object is called a client and the passed (that is, "injected") object is called a service.
hence, ObjectB
is client and ObjectA
is "injected" service.
class ObjectA {}
class ObjectB {
constructor(objectA: ObjectA);
}
which leads us to second defention
Dependency Injection is a software design concept that allows a service to be used/injected in a way that is completely independent of any client consumption.
so ObjectB
will receive the service ObjectA
without any knowledge of how the service constructed.
Dependency injection separates the creation of a client's dependencies from the client's behavior, which allows program designs to be loosely coupled.
The goal of the dependency injection technique is to remove this dependency by separating the usage from the creation of the object. This reduces the amount of required boilerplate code and improves flexibility.
With dependency injection you won't care how the service "injected object" is created or what dependencies it needs, you care only about specifing either the concrete implementation or the abstractions.
- Dependency Inversion Principle
Entities must depend on abstractions, not on concretions. It states that the high-level module must not depend on the low-level module, but they should depend on abstractions.
so to truly have loosely coupled application you should start thinking abstractions instead of using concretions directly.
class SqlConnection {
connect() {}
}
class MySqlConnection {
connect() {}
}
class DataRepository {
constructor(sqlConnection: SqlConnection) {}
async fetch() {
const connection = await this.sqlConnection.connect();
return connection.find();
}
}
const connection = new SqlConnection();
const repository = new DataRepository(connection);
this sample violates the Dependency Inversion Principle because if you thought about changing the SqlConnection
to MySqlConnection
you would have to change DataRepository
as well since it depends on SqlConnection
.
to overcome this issue you need to create abstract class for database connection
abstract class DatabaseConnection {
abstract connect(): Promise;
}
class SqlConnection extends DatabaseConnection {
connect() {}
}
class MySqlConnection extends DatabaseConnection {
connect() {}
}
class DataRepository {
constructor(databaseConnection: DatabaseConnection) {}
async fetch() {
const connection = await this.databaseConnection.connect();
return connection.find();
}
}
const sqlConnection = new SqlConnection();
const mySqlConnection = new mySqlConnection();
const repository = new DataRepository(mySqlConnection);
now, whether you choose SqlConnection
or MySqlConnection
you won't change the DataRepository
since it depends on the abstraction and not on any of the concretions.
Features
Included
- Supports Node.js and anything uses TypeScript.
- Singleton, Scoped and Transient service lifetime.
- Auto scope validation.
- Uses class constructor to identify and inject its dependencies
Not Included
- Property injection
- Injection based on name
- Custom lifetime management
TODO:
- Service disposal.
- Asynchronous registration.
- Provide user-defined options.
Replace service functionalit.Error less service registration viaTry{ServiceLifetime}
methods.- User defined extensions.
- Injector hierarchy resolver.
License
- See LICENSE
Built and maintained with ❤️ by ezzabuzaid