A simple and lightweight runtime Dependency Injection (DI) container for Kotlin Multiplatform. Ivy DI is a small dependency injection library with an intuitive API and limited features.
In a nutshell, you first register dependency factory functions in the container Di.register { SomeClass() }
and then
get instances via Di.get<SomeClass>()
. It also supports auto-wiring via autoWire(::SomeClass)
that does it
automatically for you.
interface ArticlesDataSource
class RemoteArticlesDataSource(
val client: HttpClient,
val baseUrl: BaseUrlProvider
) : ArticlesDataSource
class ArticlesRepository(val source: ArticlesDataSource)
object AppModule : Di.Module {
override fun init() = Di.appScope {
register { BaseUrlProvider("https://ivy-apps.com") }
singleton { HttpClient(CIO) }
autoWire(::RemoteArticlesDataSource)
bind<ArticlesDataSource, RemoteArticlesDataSource>()
autoWireSingleton(::ArticlesRepository)
}
}
fun main() {
Di.init(AppModule) // activates the AppModule (i.e. executes its code)
val repo = Di.get<ArticlesRepository>() // ArticlesRepository instance created
}
- Lightweight & Efficient
- One-line setup
- Simple and easy to use
- Auto-wiring
- Singletons
- Bindings
- Qualifiers
- Modules
- Scopes
- Lazy intitliazation
- KDoc on every method
Warning
Before deciding whether to use it, make sure to check the Limitations.
Based on this Benchmark test from this CI run Ivy DI performs better than Koin and Kodein on the "medium" and "complex" DI graphs test cases.
Library | Startup | Small graph | Medium graph | Complex graph |
---|---|---|---|---|
Ivy DI | โ 10โปโต ms | 0.058 ยฑ 0.001 ms | 0.204 ยฑ 0.001 ms | 1.079 ยฑ 0.004 ms |
Koin | โ 10โปยณ ms | 0.061 ยฑ 0.001 ms | 0.554 ยฑ 0.003 ms | 3.522 ยฑ 0.018 ms |
Kodein | โ 10โปโด ms | 0.051 ยฑ 0.001 ms | 0.436 ยฑ 0.002 ms | 3.725 ยฑ 0.041 ms |
We encourage developers to review the benchmark setup and propose improvements. PRs that make the benchmarks fairer and more comprehensive are more than welcome!
You can find Ivy DI in our Ivy DI Maven Central repository.
Replace ?.?.?
with the numbers from:
Gradle (Kotlin)
implementation("com.ivy-apps:di:?.?.?")
or
Version Catalog
[libraries]
ivyApps-di = { module = "com.ivy-apps:di", version = "?.?.?" }
implementation(libs.ivyApps.di)
That's all you need! Now let's start using Ivy DI โจ๏ธ
class A
class B(val a: A)
Di.appScope {
register { A() }
register { B(a = Di.get()) }
}
Instances of A
and B
won't be created until the dependencies are requested in the code (lazy creation).
// instances of B and its dependencies will be created
Di.get<B>() // instance 1 of B
Di.get<B>() // instance 2 of B
Each call to Di.get()
creates a new instance for non-singleton dependencies.
class Counter(var x: Int = 0) {
init {
print("Counter created. ")
}
}
Di.appScope {
singleton {
Counter() // instance won't be created here
}
}
println(Di.get<Counter>().x) // Counter created. 0
Di.get<Counter>().x++
println(Di.get<Counter>().x) // 1
Singleton dependencies will have only one single instance
that will be created on the first Di.get()
call.
To clear singleton instance and free memory use Di.clear(scope)
.
For example, to clear the counter - Di.clear(AppScope)
.
class A
class B(val a: A)
class C(val a: A, val b: B)
class D(val a: A, val b: B, val c: C)
Di.appScope {
autoWire(::A)
// equivalent to register { A() }
autoWireSingleton(::B) // for singletons
// equivalent to singleton { B(a = Di.get()) }
autoWire(::C)
// equivalent to register { C(Di.get(), Di.get()) }
autoWire(::D)
// equivalent to register { D(Di.get(), Di.get(), Di.get()) }
}
Di.get<D>() // instance of D created
To avoid repetitive code like register { D(Di.get(), Di.get(), Di.get()) }
it's recommended to use auto-wiring.
Tip
Always auto-wire
When possible always use auto-wiring and fallback to register/singleton only when absolutely necessary. This way you won't have to modify the registered factories when you change the constructor of the dependency that's being injected.
interface Platform
class AndroidPlatform : Platform
Di.appScope {
autoWire(::AndroidPlatform)
bind<Platform, AndroidPlatform>()
// equivalent to:
// register<Platform> { AndroidPlatform() }
}
Di.get<Platform>() // AndroidPlatform instance
To bind a specific implementation to an interface (or an abstract class) use bind<Interface, Impl>()
. Note: Impl
must be registered in the dependency graph.
interface TimeFormatter
class H24TimeFormatter : TimeFormatter
class AmPmTimeFormatter : TimeFormatter
Di.appScope {
autoWire(::H24TimeFormatter)
autoWire(::AmPmTimeFormatter)
bind<TimeFormatter, H24TimeFormatter>() // default
bind<TimeFormatter, AmPmTimeFormatter>(named = "am-pm")
}
Di.get<TimeFormatter>() // H24TimeFormatter
Di.get<TimeFormatter>(named = "am-pm") // AmPmTimeFormatter
Sometimes we need to have different instances of the same type.
To achieve this in Ivy DI, we can set qualifiers using named = "something"
(you're not limited only to strings because
named: Any
).
Important
Your "named" qualifiers must support equality checks (hashCode + equals).
object DataModule : Di.Module {
override fun init() = Di.appScope {
singleton { HttpClient(CIO) }
register { Json() }
autoWire(::LoginService)
autoWireSingleton(::AnalyticsService)
}
}
object DomainModel : Di.Module {
override fun init() = Di.app {
autoWire(::LoginUseCaseImpl)
bind<LoginUseCase, LoginUseCaseImpl>()
autoWireSingleton(::SessionManager)
}
}
Di.init(
// Registers the following modules in the DI container
DataModule,
DomainModule
)
Di.get<LoginUseCase>() // instance of LoginUseCaseImpl created
To encapsulate and re-use DI logic you can create Di.Module
.
To activate the DI module you need to call Di.init(MyModule)
.
Tip
On each architecture layer or feature layer, you can create a di
package with DI modules inside. Then in your
application root or feature entrypoint,
call Di.init(ModuleA, ModuleB, ...)
with all modules that you need to use.
Ivy DI supports grouping your dependencies into scopes. This way you can manage their lifecycle
and free resources when they are no longer needed. AppScope and FeatureScope
are built-in, but you can easily define your own scopes using Di.newScope("my-scope")
.
data class UserInfo(val id: String, val name: String)
val UserScope = Di.newScope("user")
fun Di.userScope(block: Di.Scope.() -> Unit) = Di.inScope(UserScope, block) // helper function (optional)
suspend fun login() {
val userInfo = loginInternally() // UserInfo("1", "John")
Di.userScope {
// Register dependencies for the lifecycle of a user
singleton { userInfo }
}
}
// Note: This function must be called only for logged-in users,
// otherwise Di.get() will throw an exception.
suspend fun dashboard() {
// Use user related dependencies
val userInfo = Di.get<UserInfo>()
println("Hello, ${userInfo.name}") // "Hello, John"
}
suspend fun logout() {
logoutInternally()
// Frees all dependencies in UserScope
Di.clear(UserScope) // UserInfo("1", "John") gets cleared
}
Scopes are also extremely useful for defining multiple dependencies of the same type and picking the most appropriate one based on the scope using affinity.
data class Screen(val name: String)
Di.appScope {
register<String> { "Hello from app!" }
autoWire(::Screen)
}
Di.featureScope {
register<String> { "Hello from feature!" }
autoWire(::Screen)
}
Di.get<String>(affinity = AppScope) // "Hello from app!"
Di.get<Screen>(affinity = AppScope) // Screen(name="Hello from app!")
Di.get<String>(affinity = FeatureScope) // "Hello from feature!"
Di.get<Screen>(affinity = FeatureScope) // Screen(name="Hello from feature!")
Note
Auto-wiring automatically sets the affinity to the scope from which it's called.
Currently not supported, investigating this use-case and whether we can support it nicely.
Warning
So far, we haven't found a nice and efficient solution. Currently, multi-bindings are not on the roadmap. The main blocker is the limited KClass<*> support for generics.
By default, all instances in Ivy DI are lazily initialized only after Di.get()
is called.
However, there are cases where you might want to postpone the initialization even further.
You can do that by wrapping your dependency in Lazy<T>
and using Di.getLazy<T>()
instead.
class ArticlesDataSource(val client: Lazy<HttpClient>) {
suspend fun fetchLatest(): List<Article> = client.value.get("url") // .value gets an instance of the HttpClient
}
class ArticlesRepository(val source: ArticlesDataSource) {
suspend fun fetchLatest(): List<Article> = source.fetchLatest()
}
Di.appScope {
singleton { HttpClient(CIO) }
register {
// autoWire won't work because you need to explicitly call Di.getLazy() instead of Di.get()
ArticlesDataSource(Di.getLazy())
}
autoWire(::ArticlesRepository)
}
val repo = Di.get<ArticlesRepository>() // HttpClient instance not created
repo.fetchLatest() // HttpClient instance created
The instance of HttpClient
will be created only after the ArticlesDataSource#fetchLatest()
method is called.
To avoid performance and compatibility problems we limit reflection to the bare minimum.
Ivy DI uses only KClass<*>
which unfortunately doesn't make a difference between the generic type of the class.
Problematic code:
class Container<T>(val value: T)
Di.appScope {
register {
// perceived as Container<*>, information for the Int generic type is lost
Container<Int>(42)
}
register { Container<String>("hello") } // will override the factory for Container<Int>
}
val intContainer = Di.get<Container<Int>>() // Container<String>("hello") instance created.
intContainer.value // "hello" (String) even tough we requested Container<Int>!
val x = intContainer.value + 1 // ClassCastException: class java.lang.String cannot be cast to class java.lang.Number
Di.get<Container<String>>() // Container<String>("hello") instance created
This is very weird and has some nasty implications that break type-safety.
To overcome this KClass
limitation, the workaround is to wrap your generics in value classes.
Workaround:
class Container<T>(val value: T)
@JvmInline
value class IntContainer(val value: Container<Int>)
@JvmInline
value class StringContainer(val value: Container<String>)
Di.appScope {
register { IntContainer(Container(42)) }
register { StringContainer(Container("hello")) }
}
Di.get<IntContainer>().value // Container<Int>(42) instance created
Di.get<StringContainer>().value // Container<String>("hello") instance created
The same applies for List<T>
, Set<T>
, Map<K, V>
and any generic class.
The fix is to wrap it in some value class and register + inject the value class wrapper.
The Ivy DI APIs aren't synchronized and thread-safety is a responsibility of the API user. We made this decision to keep the DI container free of Java (or 3rd party) dependencies and prioritize efficiency. We recommend registering your dependencies on the main thread.
The library will be maintained as long as Ivy Apps Ltd has an interest in using it. Given that Ivy DI currently has no community, the project may be abandoned in the future.
Disclaimer
Ivy DI is provided "as is" under the Apache 2.0 License, without warranties of any kind, including but not limited to fitness for a particular purpose or security. Use it at your own risk. The authors are not liable for any issues, defects, security vulnerabilities, or damages arising from its use. Maintenance, updates, and support are not guaranteed.
Kotlin Multiplatform projects that are using Ivy DI: