di-kit is a dependency injection toolkit for modern Go applications. It's designed to be easy-to-use, lightweight, and full-featured.
- Create the
Container
and register services using values and constructor functions. - Resolve services by type from the
Container
. - Close the
Container
when you're done. The container will callClose
on any services it created.
// 1. Create the Container and register services using values and constructor functions.
c, err := di.NewContainer(
di.WithService(logger), // var logger *slog.Logger
di.WithService(storage.NewDBStore, // NewDBStore(context.Context) (*storage.DBStore, error)
di.As[storage.Store](),
),
di.WithService(service.NewService), // NewService(*slog.Logger, storage.Store) *service.Service
)
// ...
defer func() {
// 3. Close the Container when you're done.
// The container will call Close on any services it created.
err := c.Close(ctx)
// ...
}()
// 2. Resolve services by type from the Container.
svc, err := di.Resolve[*service.Service](ctx, c)
// ...
go get github.com/sectrean/di-kit
Requires Go 1.22 or higher
Use NewContainer
on application startup to create a Container
. Register services using di.WithService()
functional options with a value or constructor function.
A value can be a struct or a pointer to a struct. When the value type is requested from the Container
, this value will be returned. The service will be registered as the value's actual type, even if the variable is declared as an interface. A service registered with a value is referred to as a value service.
A constructor function may accept any parameters. The function must return a service, and may also return an error. When the service is requested from the Container
, the function is called with the parameters resolved from the container. The service will be registered as the function's return type, which can be a struct, a pointer to a struct, or an interface. A service registered with a function is referred to as a function service.
logger := slog.New(/*...*/)
c, err := di.NewContainer(
// Value service registered as *slog.Logger
di.WithService(logger),
// Function service registered as storage.Store
di.WithService(storage.NewDBStore, di.As[storage.Store]()), // NewDBStore(context.Context) (*storage.DBStore, error)
// Function service registered as *service.Service
di.WithService(service.NewService), // NewService(*slog.Logger, storage.Store) *service.Service
)
Any errors from registering services will be joined together.
Use Resolve
to get a service from the Container
by type. If a requested service is not registered with the Container
, or a dependency cycle is detected, an error will be returned.
svc, _ := di.Resolve[*service.Service](ctx, c)
svc.Run(ctx)
Use Invoke
to invoke a function using parameters resolved from the Container
.
// var c *di.Container
err = di.Invoke(ctx, c, runService)
func runService(ctx context.Context, svc *service.Service) error {
err := svc.Start(ctx)
if err != nil {
return err
}
// Wait ...
return svc.Stop(ctx)
}
Services often need to do some clean up when they're done being used. The Container
can handle this for registered services.
On application shutdown, use Container.Close
to clean up services. By default, the Container
will call a Close
method on all services that is has created. Any errors returned from closing services will be joined together.
See Closing for more.
It's recommended that your service constructor functions accept interfaces and return structs.
By default, function services are registered as the function return type.
Use the di.As[Service]()
option to register a service as an interface that it implements. This allows your other services to depend on interfaces, which makes mocking/testing easier.
c, err := di.NewContainer(
di.WithService(storage.NewDBStore, // NewDBStore() *storage.DBStore
di.As[storage.Store](), // Register the service as implemented interface
di.As[*storage.DBStore](), // Add this if you also want to use the function return type
),
di.WithService(service.NewService), // NewService(storage.Store) *service.Service
)
By default, function services are closed with the Container
if they implement one the following Close
method signatures:
Close(context.Context) error
Close(context.Context)
Close() error
Close()
The default behavior can be disabled using the di.IgnoreClose()
option when registering the service:
c, err := di.NewContainer(
di.WithService(logger),
di.WithService(service.NewService,
// We don't want the container to automatically call Close
di.IgnoreClose(),
),
)
If a service uses another method to clean up, a custom close function can be configured using the di.WithCloseFunc()
option:
c, err := di.NewContainer(
di.WithService(logger),
di.WithService(service.NewService,
di.WithCloseFunc(func (ctx context.Context, svc *service.Service) error {
return svc.Shutdown(ctx)
}),
),
)
Value services are not closed by default since they are not created by the Container
. If you want to have the Container
close a value service, use the di.WithClose()
option to call a supported Close
method. Or use the di.WithCloseFunc()
option to specify a custom close function.
If you register multiple services as the same type, you can inject all of them as a slice, or a variadic parameter.
c, err := di.NewContainer(
di.WithService(storage.NewDBStore, di.As[storage.Store](),
di.As[healthcheck.HealthChecker](),
),
di.WithService(cache.NewRedisCache, di.As[cache.Cache](),
di.As[healthcheck.HealthChecker](),
),
// All services registered as HealthChecker will be resolved and injected as a slice
di.WithService(healthcheck.NewHealthHandler), // NewHealthHandler([]HealthChecker) *HealthHandler
)
If you want to register multiple services as the same type, but be able to differentiate them when resolving, use di.WithTag()
when registering the service.
Use di.WithTagged[Dependency]()
when registering a dependent service to specify a tag for a dependency.
c, err := di.NewContainer(
di.WithService(db.NewPrimaryDB, // NewPrimaryDB(context.Context) (*db.DB, error)
di.WithTag(db.Primary),
),
di.WithService(db.NewReplicaDB, // NewReplicaDB(context.Context) (*db.DB, error)
di.WithTag(db.Replica),
),
di.WithService(storage.NewReadWriteStore, // NewReadWriteStore(*db.DB) *storage.ReadWriteStore
di.WithTagged[*db.DB](db.Primary),
),
di.WithService(storage.NewReadOnlyStore, // NewReadOnlyStore(*db.DB) *storage.ReadOnlyStore
di.WithTagged[*db.DB](db.Replica),
),
)
// The *db.DB service tagged with db.Primary will be injected
rwStore, err := di.Resolve[*storage.ReadWriteStore](ctx, c)
// The *db.DB service tagged with db.Replica will be injected
roStore, err := di.Resolve[*storage.ReadOnlyStore](ctx, c)
Use di.WithTag()
to specify a tag when resolving a service directly from a container.
primary, err := di.Resolve[*db.DB](ctx, c, di.WithTag(db.Primary))
Lifetimes control how function services are created:
Singleton
: Only one instance of the service is created and reused every time it is resolved from the container. This is the default lifetime.Scoped
: A new instance of the service is created for each child scope of the container. See Scopes for more information.Transient
: A new instance of the service is created every time it is resolved from the container.
Specify a lifetime when registering a function service:
c, err := di.NewContainer(
di.WithService(service.NewScopedService, di.ScopedLifetime),
di.WithService(service.NewTransientService, di.TransientLifetime),
)
You can create new Containers with child scopes. Scoped dependencies can be resolved from a child scope.
c, err := di.NewContainer(
di.WithService(logger),
di.WithService(service.NewService),
di.WithService(service.NewScopedService, di.ScopedLifetime),
)
scope, err := c.NewScope()
// ...
// Don't forget to Close the scope when you're done
defer func() {
err := scope.Close(ctx)
// ...
}
New services can also be registered when creating a child scope. These new services are isolated from the parent or sibling Containers.
scope, err := c.NewScope(
di.WithService(requestService),
)
A couple services are provided directly by the container and cannot be registered.
context.Context
- When a service is resolved, the context passed into Resolve
will be injected into constructor functions as a dependency. You should avoid resolving resolving singleton services from a request-scoped context that may be canceled.
di.Scope
- The current Container
can be injected into a service as di.Scope
. This allows a service to resolve other services. The scope must be stored and only used after the constructor function returns.
func NewDBFactory(scope di.Scope) *DBFactory {
return &DBFactory{scope}
}
type DBFactory struct {
scope di.Scope
}
func (f *DBFactory) NewDB(ctx context.Context, dbName string) *DB {
// Use f.scope to resolve dependencies needed to create a *DB...
}
It's often useful to "wrap" or "decorate" a service to add some functionality.
Use di.WithDecorator()
when creating a Container
to register a decorator function.
A decorator function must accept and return a service. It may also accept other parameters which will be resolved from the container.
c, err := di.NewContainer(
di.WithService(logger), // var logger *slog.Logger
di.WithService(service.NewService, // NewService() *service.Service
di.As[service.Interface](),
),
di.WithDecorator(service.NewLoggedService), // NewLoggedService(service.Interface, *slog.Logger) service.Interface
)
// ...
svc, err := di.Resolve[service.Interface](ctx, c)
If you register multiple decorators for a service, they will be applied in the order they are registered.
Modules allow you to export a collection of container options (services, decorators, etc.) that can be re-used for different containers.
var DependencyModule = di.Module{
di.WithService(NewLogger),
//...
}
c, err := di.NewContainer(
di.WithModule(DependencyModule), // var DependencyModule di.Module
di.WithService(NewService), // NewService(*slog.Logger) *service.Service
)
The dicontext
package allows you to add a container scope to a context.Context
.
Then the scope can be retrieved from the context and used as a service locator.
// Add container scope to the context
ctx = dicontext.WithScope(ctx, c)
// Resolve services from the scope on the context
svc, err := dicontext.Resolve[*service.Service](ctx)
The dihttp
package provides configurable net/http
middleware to create new child scopes for each request. The scope is added to the request context using the dicontext
package.
c, err := di.NewContainer(
di.WithService(logger),
di.WithService(service.NewRequestService, di.ScopedLifetime), // NewRequestService(*slog.Logger, *http.Request) *service.RequestService
)
// ...
var handler http.Handler
handler = http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
// Access the scope from the request context
ctx := r.Context()
svc, err := dicontext.Resolve[*service.RequestService](ctx)
// ...
})
// Create and apply the middleware
scopeMiddleware := dihttp.RequestScopeMiddleware(c)
handler = scopeMiddleware(handler)
// ...
- Use
di.Lazy[Service any]
to inject a lazily-resolvable service. Can be used to avoid creation if service is never needed. Or to get around dependency cycles in a simpler way than injectingdi.Scope
. - Add
dicontext.WithoutScope(context.Context)
to remove/hide a scope from child contexts. - Track child scopes to make sure all child scopes have been closed. What do we do in this case? Close the child container(s)? Return an error?
- Allow retrying
Resolve
if an error was returned. Normally the first error would be cached for singleton or scoped dependencies. Subsequent attempts to resolve the service will return the error. However, there may be some cases where you would want to be able to retry the constructor function. - Implement additional Container options:
- Validate services: make sure all types are resolvable, with no cycles. (Will need to exclude scoped services in the root container since they may have dependencies registered in child scopes.)
- Automatically call
Shutdown
methods to close services. - Enable error stacktraces optionally.
- Logging with
slog
.