Depi - Dart Dependency Injection
Depi is another dependency injection library that aims to match the functionality of .NET Dependency Injection.
Table of Contents
Getting Started
Add you need to get started is create a new instance of ServiceCollection
, which is a builder class that registers all the dependencies.
After creating a new ServiceCollection
instance, start adding services:
final serviceCollection = ServiceCollection();
serviceCollection.putSingleton<HttpService>((services) => HttpService());
serviceCollection.putInstance<ServiceA>(ServiceA());
serviceCollection.putTransient<ServiceB>((services) => ServiceB());
serviceCollcetion..putTransient<ServiceD>((services) => ServiceD())
..putTransient<ServiceE>((services) => ServiceE())
..putTransient<ServiceF>((services) => ServiceF());
When finished, build the container and get an instance of ServiceProvider
.
final ServiceProvider serviceProvider = serviceCollection.build();
If you want to modify the list of services after creating the container, consider calling:
final ModifiableServiceProvider serviceProvider = serviceCollection.buildModifiable();
This will give you access to the same methods to add services you have in the ServiceCollection
class and some others to invalidate services, delete them and clear the container.
// This will delete HttpService
container.drop<HttpService>();
// This will invalidate the cached value for HttpService, if it is a lazy singleton.
container.invalidate<HttpService>();
// This will delete all the services
container.clear();
Using services
To retrieve a service, call:
final httpService = serviceProvider.service<HttpService>();
Remember that this method will throw an exception if
HttpService
is not registered. If you don't want that and instead want to receive a null value, consider usingcontainer.maybeService<HttpService>()
.
ServiceProvider
is also a callable class, so you can do:
final httpService = serviceProvider<HttpService>();
// Benefit from using type inference
final HttpService httpService = serviceProvider();
Service Types
Depi allows the creation of two service types:
- Singleton: A single instance to a type. It may be lazy, which means it will be created the next time it is requested, or a simple value, which is created when the service is registered.
- Transient: A new instance of the type will be created whenever the service is requested.
Options Pattern
The options pattern is something that comes from .NET Dependency Injection. You can read more about it here, but, in general, it provides encapsulation and separation of concerns.
For example, if you create an HttpService
you may want to allow the baseurl to be configured, so, you end up creating something like this:
HttpService({required String baseUrl}) : _baseUrl = baseUrl;
While this is not super bad, you can make it better using the options pattern. Start by creating a new class that will hold your "options" for the HttpService.
final class HttpServiceOptions {
final String baseUrl;
HttpServiceOptions({required this.baseUrl});
}
And now, update your HttpService to use the created class, but, the parameter must be of type Options<T>
, so, in this case: Options<HttpServiceOptions>
:
class HttpService {
final HttpServiceOptions _options;
HttpService({required Options<HttpServiceOptions> options}) : _options = options.value;
}
Now, to configure HttpServiceOptions
, you have a couple of options:
// Provide a fixed, instantly created, value:
serviceProvider.configureValue<HttpServiceOptions>(HttpServiceOptions(baseUrl: "https:..."));
// Provide a fixed, lazy created, value:
serviceProvider.configure<HttpServiceOptions>((services) => HttpServiceOptions(baseUrl: "https:..."));
// Provide a transient, lazily created, value. Here, a new HttpServiceOptions will be created every time it is needed.
// Keep in mind this only will work if the service that requests this is also transient.
serviceProvider.configureSnapshot<HttpServiceOptions>((services) => HttpServiceOptions(baseUrl: "https:..."));
Finally, to request an instance of HttpServiceOptions
, you call:
serviceProvider<Options<HttpServiceOptions>>();
If you are requesting it when creating a service, you can benefit from type inference and DepiContainer
being a callable class and just call:
container.registerSingleton<HttpService>((services) => HttpService(options: services()));
The same applies if you are requesting a service.
Why Options<T>
instead of just T
?
That is because you can create an OptionsStream<T>
, which notifies you every time a new T
Options is configured, not every time T
changes.
To register a new OptionsStream<T>
, call:
// Return an instance of T
serviceProvider.configureStream<T>((services) => T());
If T
extends ChangeNotifier
, Depi
will automatically listen to its changes and push a new version of T
, notifying every service that already has a version of T
.
To use it, instead of passing Options<T>
to your service constructor, pass an OptionsStream<T>
:
class HttpService {
HttpServiceOptions _options;
HttpService({required OptionsStream<HttpServiceOptions> options}) : _options = options.value {
options.onChange((newOptions) {
_options = newOptions;
});
}
}
OptionsStream
always provides an initial value you can use in your constructor. Also,value
always contains the up-to-date value.
Generating Code
If you want to generate the registration methods automatically, install the following dependencies:
flutter pub add depi dev:depi_generator dev:build_runner
Create a new container class:
part of 'my_container.depi.dart';
@serviceRegistrator
final class MyServiceProvider extends _$MyServiceProvider {}
The previous example is the minimal code needed to register services. You can also define environments creating named constructors.
Environments
To create a new environment, simply define a new named constructor:
part of 'my_container.depi.dart';
@serviceRegistrator
final class MyServiceContainer extends _$MyServiceContainer {
MyServiceContainer.development() : super._development();
MyServiceContainer.production() : super._production();
}
If you add the default constructor, you will have the 'default' environment. You can access use it in implementations using the
kDefaultEnvironment
.
Configure services
If you want to configure options for your services, override the configureServices
method and do your configureX
calls there.
Register services
To register a new service, annotate your class using either @singleton
or @transient
annotations. The implementations must be annotated
using the @Implementation()
annotation, where you can also define the environment where the implementation will be registered:
@singleton
abstract class HttpService {}
@Implementation(environments: {"development"})
final class MockHttpService extends HttpService {}
@Implementation(environments: {"production"})
final class RealHttpService extends HttpService {}
If your service class is not abstract
nor interface
and you want to change the environment where the service is registered, simply annotate the class with the Implementation
annotation too:
@singleton
@Implementation(environments: {kDefaultEnvironment, 'production'})
final class ServiceE {}
The same way you can override an implementation:
@singleton
final class ServiceB {}
@Implementation(environments: {"production"})
final class ServiceBImplementation extends ServiceB {}
If your service depends on Options
, you don't need to annotate your options class, but make sure to configure the service in the configureServices
method:
@singleton
final class ServiceC {
final ServiceCOptions options;
ServiceC({
required Options<ServiceCOptions> options,
}) : options = options.value;
}
// You need to configure this in configureServices!
final class ServiceCOptions {
final String key;
final int number;
ServiceCOptions({required this.key, required this.number});
}
More of this in the
example
folder of thedepi_generator
package
Generating code
Finally, run dart run build_runner build
to generate the code. To use the container, simply create a new instance of your ServiceRegistrator.
Using the code
Your MyServiceContainer
is an implementation of ServiceContainer
, so you can create it like so:
final serviceProvider = MyServiceContainer.production().build();
Libraries
- depi
- Depi is another dependency injection library that aims to match the functionality of .NET Dependency Injection.