From 34334396178eec3f72f1228b400cb9ec81c4ce4c Mon Sep 17 00:00:00 2001 From: Linnea Gräf Date: Tue, 7 Jan 2025 14:07:14 +0100 Subject: build: Split up dependency injection into its own package --- dependency-injection/build.gradle.kts | 8 +++ .../src/main/kotlin/moe/nea/ledger/utils/di/DI.kt | 83 ++++++++++++++++++++++ .../kotlin/moe/nea/ledger/utils/di/DIProvider.kt | 56 +++++++++++++++ .../main/kotlin/moe/nea/ledger/utils/di/Inject.kt | 6 ++ 4 files changed, 153 insertions(+) create mode 100644 dependency-injection/build.gradle.kts create mode 100644 dependency-injection/src/main/kotlin/moe/nea/ledger/utils/di/DI.kt create mode 100644 dependency-injection/src/main/kotlin/moe/nea/ledger/utils/di/DIProvider.kt create mode 100644 dependency-injection/src/main/kotlin/moe/nea/ledger/utils/di/Inject.kt (limited to 'dependency-injection') diff --git a/dependency-injection/build.gradle.kts b/dependency-injection/build.gradle.kts new file mode 100644 index 0000000..5a51941 --- /dev/null +++ b/dependency-injection/build.gradle.kts @@ -0,0 +1,8 @@ +plugins { + `java-library` + kotlin("jvm") +} + +java { + toolchain.languageVersion.set(JavaLanguageVersion.of(8)) +} diff --git a/dependency-injection/src/main/kotlin/moe/nea/ledger/utils/di/DI.kt b/dependency-injection/src/main/kotlin/moe/nea/ledger/utils/di/DI.kt new file mode 100644 index 0000000..0683063 --- /dev/null +++ b/dependency-injection/src/main/kotlin/moe/nea/ledger/utils/di/DI.kt @@ -0,0 +1,83 @@ +package moe.nea.ledger.utils.di + +import java.lang.reflect.AnnotatedElement +import java.util.Collections +import java.util.Stack + +@Suppress("UNCHECKED_CAST") +class DI { + private fun formatInjectionStack() = + injectionStack.joinToString(" -> ") + + fun getProvider(type: Class): BaseDIProvider { + val provider = providers[type] as BaseDIProvider? + ?: error("Could not find provider for type $type") + return provider + } + + private fun internalProvide(type: Class, element: AnnotatedElement? = null): T { + try { + val provider = getProvider(type) as BaseDIProvider + val context = if (element == null) provider.createEmptyContext() else provider.createContext(element) + val key = Pair(type, context) + val existingValue = values[key] + if (existingValue != null) return existingValue as T + if (type in injectionStack) { + error("Found injection cycle: ${formatInjectionStack()} -> $type") + } + injectionStack.push(type) + val value = + provider.provideWithContext(this, context) + val cycleCheckCookie = injectionStack.pop() + require(cycleCheckCookie == type) { "Unbalanced stack cookie: $cycleCheckCookie != $type" } + values[key] = value + return value + } catch (ex: Exception) { + throw RuntimeException("Could not create instance for type $type (in stack ${formatInjectionStack()})", ex) + } + } + + fun provide(type: Class, element: AnnotatedElement? = null): T { + return internalProvide(type, element) + } + + inline fun provide(): T = provide(T::class.java) + + fun register(type: Class, provider: BaseDIProvider) { + providers[type] = provider + } + + fun registerInjectableInterface(parent: Class, type: Class) { + internalRegisterInjectableClass(type) + register(parent, DIProvider.fromInheritance(type)) + } + + fun registerInjectableClasses(vararg type: Class<*>) { + type.forEach { internalRegisterInjectableClass(it) } + } + + private fun internalRegisterInjectableClass(type: Class) { + register(type, DIProvider.fromInjectableClass(type)) + } + + fun instantiateAll() { + providers.keys.forEach { + provide(it, null) + } + } + + fun getAllInstances(): Collection = + Collections.unmodifiableCollection(values.values) + + fun registerSingleton(value: T) { + register(value.javaClass, DIProvider.singeleton(value)) + } + + private val injectionStack: Stack> = Stack() + private val values = mutableMapOf, *>, Any>() + private val providers = mutableMapOf, BaseDIProvider<*, *>>() + + init { + registerSingleton(this) + } +} \ No newline at end of file diff --git a/dependency-injection/src/main/kotlin/moe/nea/ledger/utils/di/DIProvider.kt b/dependency-injection/src/main/kotlin/moe/nea/ledger/utils/di/DIProvider.kt new file mode 100644 index 0000000..8a54d5f --- /dev/null +++ b/dependency-injection/src/main/kotlin/moe/nea/ledger/utils/di/DIProvider.kt @@ -0,0 +1,56 @@ +package moe.nea.ledger.utils.di + +import java.lang.reflect.AnnotatedElement +import java.lang.reflect.Constructor + +fun interface DIProvider : BaseDIProvider { + override fun provideWithContext(di: DI, context: Unit): T { + return provide(di) + } + + override fun createContext(element: AnnotatedElement) { + } + + override fun createEmptyContext() { + } + + fun provide(di: DI): T + + companion object { + + fun fromInjectableClass(clazz: Class): DIProvider { + @Suppress("UNCHECKED_CAST") + val cons = (clazz.constructors.find { it.getAnnotation(Inject::class.java) != null } + ?: clazz.constructors.find { it.parameterCount == 0 } + ?: error("Could not find DI injection entrypoint for class $clazz")) + as Constructor + // TODO: consider using unsafe init to inject the parameters *before* calling the constructor + return DIProvider { di -> + val typArgs = cons.parameters.map { + di.provide(it.type, it) + }.toTypedArray() + val instance = cons.newInstance(*typArgs) + for (it in clazz.fields) { + if (it.getAnnotation(Inject::class.java) == null) continue + it.set(instance, di.provide(it.type, it)) + } + instance + } + } + + fun singeleton(value: T): DIProvider { + return DIProvider { _ -> value } + } + + fun fromInheritance(type: Class): DIProvider { + return DIProvider { it.provide(type) } + } + } + +} + +interface BaseDIProvider { + fun createContext(element: AnnotatedElement): C + fun provideWithContext(di: DI, context: C): T + fun createEmptyContext(): C +} diff --git a/dependency-injection/src/main/kotlin/moe/nea/ledger/utils/di/Inject.kt b/dependency-injection/src/main/kotlin/moe/nea/ledger/utils/di/Inject.kt new file mode 100644 index 0000000..a8fdd87 --- /dev/null +++ b/dependency-injection/src/main/kotlin/moe/nea/ledger/utils/di/Inject.kt @@ -0,0 +1,6 @@ +package moe.nea.ledger.utils.di + +@Retention(AnnotationRetention.RUNTIME) +@Target(AnnotationTarget.CONSTRUCTOR, AnnotationTarget.FIELD) +annotation class Inject( +) -- cgit