mirror of
				https://github.com/InsanusMokrassar/MicroUtils.git
				synced 2025-10-26 09:40:26 +00:00 
			
		
		
		
	Compare commits
	
		
			48 Commits
		
	
	
		
			revert-245
			...
			v0.19.5
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| c6785f1a4f | |||
| 83fe621c56 | |||
| b3a93e17eb | |||
| 546a391af3 | |||
| 786cf9bd8b | |||
| dfd6fe062d | |||
| b6ef818613 | |||
| b0f9e9c30a | |||
| 7e5c88ddc3 | |||
| 9824c3e00f | |||
| 9171d5ed11 | |||
| a1830ebb82 | |||
| 22d2a3d9bf | |||
| c9b97fc965 | |||
| 51f85becd5 | |||
| a8a281cfb4 | |||
| ddd1304949 | |||
| 86d70b6c02 | |||
| a22bdb39e7 | |||
| 7ae4d5ef95 | |||
| a2038cbefa | |||
| 992091eade | |||
| e3bfead0c5 | |||
| 0de96141fd | |||
| fa18d15c3c | |||
| ea9dbf2371 | |||
| d34e3ec7a9 | |||
| c8833a36af | |||
| a067cb0c0f | |||
| 999c8327bd | |||
| c2ec73c70a | |||
| 702f782fc1 | |||
| 25dbcaaf83 | |||
| b00d454a24 | |||
| dbc921d56d | |||
| 438fefa7a3 | |||
| 5d74eac814 | |||
| 9fb62e1e25 | |||
| 3e366ea73b | |||
| 0ff895bffa | |||
| c5bb120280 | |||
| 4b26a92b37 | |||
| 0a8453b4d2 | |||
| c9872a61b6 | |||
| 149a1aa278 | |||
| 13d0e1b682 | |||
| 6f9be2a9f8 | |||
| 93ba98d993 | 
							
								
								
									
										65
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										65
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -1,5 +1,70 @@ | ||||
| # Changelog | ||||
|  | ||||
| ## 0.19.5 | ||||
|  | ||||
| * `Repos`: | ||||
|     * `Generator`: | ||||
|         * Fixes in new type generation | ||||
|  | ||||
| ## 0.19.4 | ||||
|  | ||||
| * `Versions`: | ||||
|     * `Koin`: `3.4.1` -> `3.4.2` | ||||
|     * `Android Fragments`: `1.5.7` -> `1.6.0` | ||||
| * `Koin` | ||||
|     * `Generator` | ||||
|         * Fixes in new generic generator part | ||||
|  | ||||
| ## 0.19.3 | ||||
|  | ||||
| * `Koin` | ||||
|     * `Generator` | ||||
|         * New getter methods now available with opportunity to use parameters | ||||
|         * Old notation `*Single` and `*Factory` is deprecated since this release. With old | ||||
|           will be generated new `single*` and `factory*` notations for new generations | ||||
|         * Add opportunity to use generic-oriented koin definitions | ||||
|  | ||||
| ## 0.19.2 | ||||
|  | ||||
| * `Versions`: | ||||
|     * `Ktor`: `2.3.0` -> `2.3.1` | ||||
|     * `Koin`: `3.4.0` -> `3.4.1` | ||||
|     * `Uuid`: `0.7.0` -> `0.7.1` | ||||
|  | ||||
| ## 0.19.1 | ||||
|  | ||||
| * `Versions`: | ||||
|     * `Korlibs`: `4.0.1` -> `4.0.3` | ||||
|     * `Kotlin Poet`: `1.13.2` -> `1.14.0` | ||||
|  | ||||
| ## 0.19.0 | ||||
|  | ||||
| * `Versions`: | ||||
|     * `Korlibs`: `3.4.0` -> `4.0.1` | ||||
|  | ||||
| ## 0.18.4 | ||||
|  | ||||
| * `Koin`: | ||||
|     * New extension `lazyInject` | ||||
|  | ||||
| ## 0.18.3 | ||||
|  | ||||
| * `Versions`: | ||||
|     * `Serialization`: `1.5.0` -> `1.5.1` | ||||
|     * `Android Cor Ktx`: `1.10.0` -> `1.10.1` | ||||
|  | ||||
| ## 0.18.2 | ||||
|  | ||||
| * `Startup`: | ||||
|     * Now internal `Json` is fully customizable | ||||
|  | ||||
| ## 0.18.1 | ||||
|  | ||||
| * `Common`: | ||||
|     * Add `MapDiff` | ||||
| * `Coroutines`: | ||||
|     * Add `SmartMutex` | ||||
|  | ||||
| ## 0.18.0 | ||||
|  | ||||
| **ALL PREVIOUSLY DEPRECATED FUNCTIONALITY HAVE BEEN REMOVED** | ||||
|   | ||||
| @@ -200,20 +200,18 @@ inline fun <T> Iterable<T>.calculateStrictDiff( | ||||
| ) = calculateDiff(other, strictComparison = true) | ||||
|  | ||||
| /** | ||||
|  * This method call [calculateDiff] with strict mode [strictComparison] and then apply differences to [this] | ||||
|  * mutable list | ||||
|  * Applies [diff] to [this] [MutableList] | ||||
|  */ | ||||
| fun <T> MutableList<T>.applyDiff( | ||||
|     source: Iterable<T>, | ||||
|     strictComparison: Boolean = false | ||||
| ): Diff<T> = calculateDiff(source, strictComparison).also { | ||||
|     for (i in it.removed.indices.sortedDescending()) { | ||||
|         removeAt(it.removed[i].index) | ||||
|     diff: Diff<T> | ||||
| ) { | ||||
|     for (i in diff.removed.indices.sortedDescending()) { | ||||
|         removeAt(diff.removed[i].index) | ||||
|     } | ||||
|     it.added.forEach { (i, t) -> | ||||
|     diff.added.forEach { (i, t) -> | ||||
|         add(i, t) | ||||
|     } | ||||
|     it.replaced.forEach { (_, new) -> | ||||
|     diff.replaced.forEach { (_, new) -> | ||||
|         set(new.index, new.value) | ||||
|     } | ||||
| } | ||||
| @@ -222,17 +220,30 @@ fun <T> MutableList<T>.applyDiff( | ||||
|  * This method call [calculateDiff] with strict mode [strictComparison] and then apply differences to [this] | ||||
|  * mutable list | ||||
|  */ | ||||
| fun <T> MutableList<T>.applyDiff( | ||||
|     source: Iterable<T>, | ||||
|     strictComparison: Boolean = false | ||||
| ): Diff<T> = calculateDiff(source, strictComparison).also { | ||||
|     applyDiff(it) | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This method call [calculateDiff] and then apply differences to [this] | ||||
|  * mutable list | ||||
|  */ | ||||
| fun <T> MutableList<T>.applyDiff( | ||||
|     source: Iterable<T>, | ||||
|     comparisonFun: (T?, T?) -> Boolean | ||||
| ): Diff<T> = calculateDiff(source, comparisonFun).also { | ||||
|     for (i in it.removed.indices.sortedDescending()) { | ||||
|         removeAt(it.removed[i].index) | ||||
|     } | ||||
|     it.added.forEach { (i, t) -> | ||||
|         add(i, t) | ||||
|     } | ||||
|     it.replaced.forEach { (_, new) -> | ||||
|         set(new.index, new.value) | ||||
|     } | ||||
|     applyDiff(it) | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Reverse [this] [Diff]. Result will contain [Diff.added] on [Diff.removed] (and vice-verse), all the | ||||
|  * [Diff.replaced] values will be reversed too | ||||
|  */ | ||||
| fun <T> Diff<T>.reversed() = Diff( | ||||
|     removed = added, | ||||
|     replaced = replaced.map { it.second to it.first }, | ||||
|     added = removed | ||||
| ) | ||||
|   | ||||
| @@ -0,0 +1,135 @@ | ||||
| package dev.inmo.micro_utils.common | ||||
|  | ||||
| /** | ||||
|  * Contains diff based on the comparison of objects with the same [K]. | ||||
|  * | ||||
|  * @param removed Contains map with keys removed from parent map | ||||
|  * @param changed Contains map with keys values changed new map in comparison with old one | ||||
|  * @param added Contains map with new keys and values | ||||
|  */ | ||||
| data class MapDiff<K, V> @Warning(warning) constructor( | ||||
|     val removed: Map<K, V>, | ||||
|     val changed: Map<K, Pair<V, V>>, | ||||
|     val added: Map<K, V> | ||||
| ) { | ||||
|     fun isEmpty() = removed.isEmpty() && changed.isEmpty() && added.isEmpty() | ||||
|     inline fun isNotEmpty() = !isEmpty() | ||||
|  | ||||
|     companion object { | ||||
|         private const val warning = "This feature can be changed without any warranties. Use with caution and only in case you know what you are doing" | ||||
|         fun <K, V> empty() = MapDiff<K, V>(emptyMap(), emptyMap(), emptyMap()) | ||||
|     } | ||||
| } | ||||
|  | ||||
| private inline fun <K, V> createCompareFun( | ||||
|     strictComparison: Boolean | ||||
| ): (K, V, V) -> Boolean = if (strictComparison) { | ||||
|     { _, first, second -> first === second } | ||||
| } else { | ||||
|     { _, first, second -> first == second } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Compare [this] [Map] with the [other] one in principle when [other] is newer than [this] | ||||
|  * | ||||
|  * @param compareFun Will be used to determine changed values | ||||
|  */ | ||||
| fun <K, V> Map<K, V>.diff( | ||||
|     other: Map<K, V>, | ||||
|     compareFun: (K, V, V) -> Boolean | ||||
| ): MapDiff<K, V> { | ||||
|     val removed: Map<K, V> = (keys - other.keys).associateWith { | ||||
|         getValue(it) | ||||
|     } | ||||
|     val added: Map<K, V> = (other.keys - keys).associateWith { | ||||
|         other.getValue(it) | ||||
|     } | ||||
|     val changed = keys.intersect(other.keys).mapNotNull { | ||||
|         val old = getValue(it) | ||||
|         val new = other.getValue(it) | ||||
|         if (compareFun(it, old, new)) { | ||||
|             return@mapNotNull null | ||||
|         } else { | ||||
|             it to (old to new) | ||||
|         } | ||||
|     }.toMap() | ||||
|  | ||||
|     return MapDiff( | ||||
|         removed, | ||||
|         changed, | ||||
|         added | ||||
|     ) | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Compare [this] [Map] with the [other] one in principle when [other] is newer than [this] | ||||
|  * | ||||
|  * @param strictComparison If true, will use strict (===) comparison for the values' comparison. Otherwise, standard | ||||
|  * `equals` will be used | ||||
|  */ | ||||
| fun <K, V> Map<K, V>.diff( | ||||
|     other: Map<K, V>, | ||||
|     strictComparison: Boolean = false | ||||
| ): MapDiff<K, V> = diff( | ||||
|     other, | ||||
|     compareFun = createCompareFun(strictComparison) | ||||
| ) | ||||
|  | ||||
| /** | ||||
|  * Will apply [mapDiff] to [this] [MutableMap] | ||||
|  */ | ||||
| fun <K, V> MutableMap<K, V>.applyDiff( | ||||
|     mapDiff: MapDiff<K, V> | ||||
| ) { | ||||
|     mapDiff.apply { | ||||
|         removed.keys.forEach { remove(it) } | ||||
|         changed.forEach { (k, oldNew) -> | ||||
|             put(k, oldNew.second) | ||||
|         } | ||||
|         added.forEach { (k, new) -> | ||||
|             put(k, new) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Will apply changes with [from] map into [this] one | ||||
|  * | ||||
|  * @param compareFun Will be used to determine changed values | ||||
|  * | ||||
|  * @return [MapDiff] applied to [this] [MutableMap] | ||||
|  */ | ||||
| fun <K, V> MutableMap<K, V>.applyDiff( | ||||
|     from: Map<K, V>, | ||||
|     compareFun: (K, V, V) -> Boolean | ||||
| ): MapDiff<K, V> { | ||||
|     return diff(from, compareFun).also { | ||||
|         applyDiff(it) | ||||
|     } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Will apply changes with [from] map into [this] one | ||||
|  * | ||||
|  * @param strictComparison If true, will use strict (===) comparison for the values' comparison. Otherwise, standard | ||||
|  * `equals` will be used | ||||
|  * | ||||
|  * @return [MapDiff] applied to [this] [MutableMap] | ||||
|  */ | ||||
| fun <K, V> MutableMap<K, V>.applyDiff( | ||||
|     from: Map<K, V>, | ||||
|     strictComparison: Boolean = false | ||||
| ): MapDiff<K, V> = applyDiff( | ||||
|     from, | ||||
|     compareFun = createCompareFun(strictComparison) | ||||
| ) | ||||
|  | ||||
| /** | ||||
|  * Reverse [this] [MapDiff]. Result will contain [MapDiff.added] on [MapDiff.removed] (and vice-verse), all the | ||||
|  * [MapDiff.changed] values will be reversed too | ||||
|  */ | ||||
| fun <K, V> MapDiff<K, V>.reversed(): MapDiff<K, V> = MapDiff( | ||||
|     removed = added, | ||||
|     changed = changed.mapValues { (_, oldNew) -> oldNew.second to oldNew.first }, | ||||
|     added = removed | ||||
| ) | ||||
| @@ -0,0 +1,136 @@ | ||||
| package dev.inmo.micro_utils.coroutines | ||||
|  | ||||
| import kotlinx.coroutines.currentCoroutineContext | ||||
| import kotlinx.coroutines.flow.MutableStateFlow | ||||
| import kotlinx.coroutines.flow.StateFlow | ||||
| import kotlinx.coroutines.flow.asStateFlow | ||||
| import kotlinx.coroutines.flow.first | ||||
| import kotlinx.coroutines.isActive | ||||
| import kotlinx.coroutines.sync.Mutex | ||||
| import kotlinx.coroutines.sync.withLock | ||||
| import kotlin.contracts.ExperimentalContracts | ||||
| import kotlin.contracts.InvocationKind | ||||
| import kotlin.contracts.contract | ||||
|  | ||||
| /** | ||||
|  * It is interface which will work like classic [Mutex], but in difference have [lockStateFlow] for listening of the | ||||
|  * [SmartMutex] state. | ||||
|  * | ||||
|  * There is [Mutable] and [Immutable] realizations. In case you are owner and manager current state of lock, you need | ||||
|  * [Mutable] [SmartMutex]. Otherwise, [Immutable]. | ||||
|  * | ||||
|  * Any [Mutable] [SmartMutex] may produce its [Immutable] variant which will contains [lockStateFlow] equal to its | ||||
|  * [Mutable] creator | ||||
|  */ | ||||
| sealed interface SmartMutex { | ||||
|     val lockStateFlow: StateFlow<Boolean> | ||||
|  | ||||
|     /** | ||||
|      * * True - locked | ||||
|      * * False - unlocked | ||||
|      */ | ||||
|     val isLocked: Boolean | ||||
|         get() = lockStateFlow.value | ||||
|  | ||||
|     /** | ||||
|      * Immutable variant of [SmartMutex]. In fact will depend on the owner of [lockStateFlow] | ||||
|      */ | ||||
|     class Immutable(override val lockStateFlow: StateFlow<Boolean>) : SmartMutex | ||||
|  | ||||
|     /** | ||||
|      * Mutable variant of [SmartMutex]. With that variant you may [lock] and [unlock]. Besides, you may create | ||||
|      * [Immutable] variant of [this] instance with [immutable] factory | ||||
|      * | ||||
|      * @param locked Preset state of [isLocked] and its internal [_lockStateFlow] | ||||
|      */ | ||||
|     class Mutable(locked: Boolean = false) : SmartMutex { | ||||
|         private val _lockStateFlow = MutableStateFlow<Boolean>(locked) | ||||
|         override val lockStateFlow: StateFlow<Boolean> = _lockStateFlow.asStateFlow() | ||||
|  | ||||
|         private val internalChangesMutex = Mutex() | ||||
|  | ||||
|         fun immutable() = Immutable(lockStateFlow) | ||||
|  | ||||
|         /** | ||||
|          * Holds call until this [SmartMutex] will be re-locked. That means that while [isLocked] == true, [holds] will | ||||
|          * wait for [isLocked] == false and then try to lock | ||||
|          */ | ||||
|         suspend fun lock() { | ||||
|             do { | ||||
|                 waitUnlock() | ||||
|                 val shouldContinue = internalChangesMutex.withLock { | ||||
|                     if (_lockStateFlow.value) { | ||||
|                         true | ||||
|                     } else { | ||||
|                         _lockStateFlow.value = true | ||||
|                         false | ||||
|                     } | ||||
|                 } | ||||
|             } while (shouldContinue && currentCoroutineContext().isActive) | ||||
|         } | ||||
|  | ||||
|         /** | ||||
|          * Will try to lock this [SmartMutex] immediataly | ||||
|          * | ||||
|          * @return True if lock was successful. False otherwise | ||||
|          */ | ||||
|         suspend fun tryLock(): Boolean { | ||||
|             return if (!_lockStateFlow.value) { | ||||
|                 internalChangesMutex.withLock { | ||||
|                     if (!_lockStateFlow.value) { | ||||
|                         _lockStateFlow.value = true | ||||
|                         true | ||||
|                     } else { | ||||
|                         false | ||||
|                     } | ||||
|                 } | ||||
|             } else { | ||||
|                 false | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         /** | ||||
|          * If [isLocked] == true - will change it to false and return true. If current call will not unlock this | ||||
|          * [SmartMutex] - false | ||||
|          */ | ||||
|         suspend fun unlock(): Boolean { | ||||
|             return if (_lockStateFlow.value) { | ||||
|                 internalChangesMutex.withLock { | ||||
|                     if (_lockStateFlow.value) { | ||||
|                         _lockStateFlow.value = false | ||||
|                         true | ||||
|                     } else { | ||||
|                         false | ||||
|                     } | ||||
|                 } | ||||
|             } else { | ||||
|                 false | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Will call [SmartMutex.Mutable.lock], then execute [action] and return the result after [SmartMutex.Mutable.unlock] | ||||
|  */ | ||||
| @OptIn(ExperimentalContracts::class) | ||||
| suspend inline fun <T> SmartMutex.Mutable.withLock(action: () -> T): T { | ||||
|     contract { | ||||
|         callsInPlace(action, InvocationKind.EXACTLY_ONCE) | ||||
|     } | ||||
|  | ||||
|     lock() | ||||
|     try { | ||||
|         return action() | ||||
|     } finally { | ||||
|         unlock() | ||||
|     } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Will wait until the [SmartMutex.lockStateFlow] of [this] instance will be false. | ||||
|  * | ||||
|  * Anyway, after the end of this block there are no any guaranties that [SmartMutex.isLocked] == false due to the fact | ||||
|  * that some other parties may lock it again | ||||
|  */ | ||||
| suspend fun SmartMutex.waitUnlock() = lockStateFlow.first { !it } | ||||
| @@ -1,6 +1,6 @@ | ||||
| package dev.inmo.micro_utils.crypto | ||||
|  | ||||
| import com.soywiz.krypto.md5 | ||||
| import korlibs.crypto.md5 | ||||
|  | ||||
| typealias MD5 = String | ||||
|  | ||||
|   | ||||
| @@ -13,10 +13,10 @@ repositories { | ||||
|  | ||||
| kotlin { | ||||
|     jvm() | ||||
| //    js(IR) { | ||||
| //        browser() | ||||
| //        nodejs() | ||||
| //    } | ||||
|     js(IR) { | ||||
|         browser() | ||||
|         nodejs() | ||||
|     } | ||||
|     android {} | ||||
|  | ||||
|     sourceSets { | ||||
| @@ -26,44 +26,44 @@ kotlin { | ||||
|  | ||||
|                 project.parent.subprojects.forEach { | ||||
|                     if ( | ||||
|                         it != project | ||||
|                         && it.hasProperty("kotlin") | ||||
|                         && it.kotlin.sourceSets.any { it.name.contains("commonMain") } | ||||
| //                        && it.kotlin.sourceSets.any { it.name.contains("jsMain") } | ||||
|                         && it.kotlin.sourceSets.any { it.name.contains("jvmMain") } | ||||
|                         && it.kotlin.sourceSets.any { it.name.contains("androidMain") } | ||||
|                             it != project | ||||
|                                     && it.hasProperty("kotlin") | ||||
|                                     && it.kotlin.sourceSets.any { it.name.contains("commonMain") } | ||||
|                                     && it.kotlin.sourceSets.any { it.name.contains("jsMain") } | ||||
|                                     && it.kotlin.sourceSets.any { it.name.contains("jvmMain") } | ||||
|                                     && it.kotlin.sourceSets.any { it.name.contains("androidMain") } | ||||
|                     ) { | ||||
|                         api it | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
| //        jsMain { | ||||
| //            dependencies { | ||||
| //                implementation kotlin('stdlib') | ||||
|         jsMain { | ||||
|             dependencies { | ||||
|                 implementation kotlin('stdlib') | ||||
|  | ||||
| //                project.parent.subprojects.forEach { | ||||
| //                    if ( | ||||
| //                        it != project | ||||
| //                        && it.hasProperty("kotlin") | ||||
| //                        && it.kotlin.sourceSets.any { it.name.contains("commonMain") } | ||||
| //                        && it.kotlin.sourceSets.any { it.name.contains("jsMain") } | ||||
| //                    ) { | ||||
| //                        api it | ||||
| //                    } | ||||
| //                } | ||||
| //            } | ||||
| //        } | ||||
|                 project.parent.subprojects.forEach { | ||||
|                     if ( | ||||
|                             it != project | ||||
|                                     && it.hasProperty("kotlin") | ||||
|                                     && it.kotlin.sourceSets.any { it.name.contains("commonMain") } | ||||
|                                     && it.kotlin.sourceSets.any { it.name.contains("jsMain") } | ||||
|                     ) { | ||||
|                         api it | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         jvmMain { | ||||
|             dependencies { | ||||
|                 implementation kotlin('stdlib') | ||||
|  | ||||
|                 project.parent.subprojects.forEach { | ||||
|                     if ( | ||||
|                         it != project | ||||
|                         && it.hasProperty("kotlin") | ||||
|                         && it.kotlin.sourceSets.any { it.name.contains("commonMain") } | ||||
|                         && it.kotlin.sourceSets.any { it.name.contains("jvmMain") } | ||||
|                             it != project | ||||
|                                     && it.hasProperty("kotlin") | ||||
|                                     && it.kotlin.sourceSets.any { it.name.contains("commonMain") } | ||||
|                                     && it.kotlin.sourceSets.any { it.name.contains("jvmMain") } | ||||
|                     ) { | ||||
|                         api it | ||||
|                     } | ||||
| @@ -76,10 +76,10 @@ kotlin { | ||||
|  | ||||
|                 project.parent.subprojects.forEach { | ||||
|                     if ( | ||||
|                         it != project | ||||
|                         && it.hasProperty("kotlin") | ||||
|                         && it.kotlin.sourceSets.any { it.name.contains("commonMain") } | ||||
|                         && it.kotlin.sourceSets.any { it.name.contains("androidMain") } | ||||
|                             it != project | ||||
|                                     && it.hasProperty("kotlin") | ||||
|                                     && it.kotlin.sourceSets.any { it.name.contains("commonMain") } | ||||
|                                     && it.kotlin.sourceSets.any { it.name.contains("androidMain") } | ||||
|                     ) { | ||||
|                         api it | ||||
|                     } | ||||
| @@ -116,9 +116,9 @@ tasks.dokkaHtml { | ||||
|             sourceRoots.setFrom(findSourcesWithName("commonMain")) | ||||
|         } | ||||
|  | ||||
| //        named("jsMain") { | ||||
| //            sourceRoots.setFrom(findSourcesWithName("jsMain", "commonMain")) | ||||
| //        } | ||||
|         named("jsMain") { | ||||
|             sourceRoots.setFrom(findSourcesWithName("jsMain")) | ||||
|         } | ||||
|  | ||||
|         named("jvmMain") { | ||||
|             sourceRoots.setFrom(findSourcesWithName("jvmMain")) | ||||
|   | ||||
| @@ -14,5 +14,5 @@ crypto_js_version=4.1.1 | ||||
| # Project data | ||||
|  | ||||
| group=dev.inmo | ||||
| version=0.18.0 | ||||
| android_code_version=191 | ||||
| version=0.19.5 | ||||
| android_code_version=201 | ||||
|   | ||||
| @@ -1,36 +1,36 @@ | ||||
| [versions] | ||||
|  | ||||
| kt = "1.8.20" | ||||
| kt-serialization = "1.5.0" | ||||
| kt-serialization = "1.5.1" | ||||
| kt-coroutines = "1.6.4" | ||||
|  | ||||
| kslog = "1.1.1" | ||||
|  | ||||
| jb-compose = "1.4.0" | ||||
| jb-exposed = "0.41.1" | ||||
| jb-dokka = "1.8.10" | ||||
| jb-dokka = "1.8.20" | ||||
|  | ||||
| korlibs = "3.4.0" | ||||
| uuid = "0.7.0" | ||||
| korlibs = "4.0.3" | ||||
| uuid = "0.7.1" | ||||
|  | ||||
| ktor = "2.3.0" | ||||
| ktor = "2.3.1" | ||||
|  | ||||
| gh-release = "2.4.1" | ||||
|  | ||||
| koin = "3.4.0" | ||||
| koin = "3.4.2" | ||||
|  | ||||
| okio = "3.3.0" | ||||
|  | ||||
| ksp = "1.8.20-1.0.11" | ||||
| kotlin-poet = "1.13.0" | ||||
| kotlin-poet = "1.14.2" | ||||
|  | ||||
| android-gradle = "7.4.2" | ||||
| dexcount = "4.0.0" | ||||
|  | ||||
| android-coreKtx = "1.10.0" | ||||
| android-coreKtx = "1.10.1" | ||||
| android-recyclerView = "1.3.0" | ||||
| android-appCompat = "1.6.1" | ||||
| android-fragment = "1.5.7" | ||||
| android-fragment = "1.6.0" | ||||
| android-espresso = "3.5.1" | ||||
| android-test = "1.1.5" | ||||
|  | ||||
|   | ||||
| @@ -13,3 +13,8 @@ dependencies { | ||||
|     api libs.kotlin.poet | ||||
|     api libs.ksp | ||||
| } | ||||
|  | ||||
| java { | ||||
|     sourceCompatibility = JavaVersion.VERSION_1_8 | ||||
|     targetCompatibility = JavaVersion.VERSION_1_8 | ||||
| } | ||||
|   | ||||
| @@ -9,19 +9,28 @@ import com.google.devtools.ksp.processing.Resolver | ||||
| import com.google.devtools.ksp.processing.SymbolProcessor | ||||
| import com.google.devtools.ksp.symbol.KSAnnotated | ||||
| import com.google.devtools.ksp.symbol.KSFile | ||||
| import com.google.devtools.ksp.symbol.Modifier | ||||
| import com.squareup.kotlinpoet.AnnotationSpec | ||||
| import com.squareup.kotlinpoet.ClassName | ||||
| import com.squareup.kotlinpoet.CodeBlock | ||||
| import com.squareup.kotlinpoet.FileSpec | ||||
| import com.squareup.kotlinpoet.FunSpec | ||||
| import com.squareup.kotlinpoet.KModifier | ||||
| import com.squareup.kotlinpoet.ParameterSpec | ||||
| import com.squareup.kotlinpoet.ParameterizedTypeName | ||||
| import com.squareup.kotlinpoet.ParameterizedTypeName.Companion.parameterizedBy | ||||
| import com.squareup.kotlinpoet.PropertySpec | ||||
| import com.squareup.kotlinpoet.TypeName | ||||
| import com.squareup.kotlinpoet.TypeVariableName | ||||
| import com.squareup.kotlinpoet.asTypeName | ||||
| import com.squareup.kotlinpoet.ksp.toClassName | ||||
| import com.squareup.kotlinpoet.ksp.toTypeName | ||||
| import com.squareup.kotlinpoet.ksp.writeTo | ||||
| import dev.inmo.micro_utils.koin.annotations.GenerateGenericKoinDefinition | ||||
| import dev.inmo.micro_utils.koin.annotations.GenerateKoinDefinition | ||||
| import org.koin.core.Koin | ||||
| import org.koin.core.module.Module | ||||
| import org.koin.core.parameter.ParametersDefinition | ||||
| import org.koin.core.scope.Scope | ||||
| import java.io.File | ||||
| import kotlin.reflect.KClass | ||||
| @@ -32,11 +41,236 @@ class Processor( | ||||
|     private val definitionClassName = ClassName("org.koin.core.definition", "Definition") | ||||
|     private val koinDefinitionClassName = ClassName("org.koin.core.definition", "KoinDefinition") | ||||
|  | ||||
|     private fun FileSpec.Builder.addCodeForType( | ||||
|         targetType: TypeName, | ||||
|         name: String, | ||||
|         nullable: Boolean, | ||||
|         generateSingle: Boolean, | ||||
|         generateFactory: Boolean, | ||||
|     ) { | ||||
|         val targetTypeAsGenericType = (targetType as? TypeVariableName) ?.copy(reified = true) | ||||
|  | ||||
|         fun addGetterProperty( | ||||
|             receiver: KClass<*> | ||||
|         ) { | ||||
|             addProperty( | ||||
|                 PropertySpec.builder( | ||||
|                     name, | ||||
|                     targetType, | ||||
|                 ).apply { | ||||
|                     addKdoc( | ||||
|                         """ | ||||
|                             @return Definition by key "${name}" | ||||
|                         """.trimIndent() | ||||
|                     ) | ||||
|                     getter( | ||||
|                         FunSpec.getterBuilder().apply { | ||||
|                             targetTypeAsGenericType ?.let { | ||||
|                                 addModifiers(KModifier.INLINE) | ||||
|                             } | ||||
|                             addCode( | ||||
|                                 "return " + (if (nullable) { | ||||
|                                     "getOrNull" | ||||
|                                 } else { | ||||
|                                     "get" | ||||
|                                 }) + "(named(\"${name}\"))" | ||||
|                             ) | ||||
|                         }.build() | ||||
|                     ) | ||||
|                     targetTypeAsGenericType ?.let { | ||||
|                         addTypeVariable(it) | ||||
|                     } | ||||
|                     receiver(receiver) | ||||
|                 }.build() | ||||
|             ) | ||||
|         } | ||||
|  | ||||
|         if (targetTypeAsGenericType == null) { | ||||
|             addGetterProperty(Scope::class) | ||||
|             addGetterProperty(Koin::class) | ||||
|         } | ||||
|  | ||||
|         val parametersDefinitionClassName = ClassName( | ||||
|             "org.koin.core.parameter", | ||||
|             "ParametersDefinition" | ||||
|         ) | ||||
|         fun addGetterMethod( | ||||
|             receiver: KClass<*> | ||||
|         ) { | ||||
|             addFunction( | ||||
|                 FunSpec.builder( | ||||
|                     name | ||||
|                 ).apply { | ||||
|                     addKdoc( | ||||
|                         """ | ||||
|                             @return Definition by key "${name}" with [parameters] | ||||
|                         """.trimIndent() | ||||
|                     ) | ||||
|                     receiver(receiver) | ||||
|                     addParameter( | ||||
|                         ParameterSpec( | ||||
|                             "parameters", | ||||
|                             parametersDefinitionClassName.let { | ||||
|                                 if (targetTypeAsGenericType != null) { | ||||
|                                     it.copy(nullable = true) | ||||
|                                 } else { | ||||
|                                     it | ||||
|                                 } | ||||
|                             }, | ||||
|                             KModifier.NOINLINE | ||||
|                         ).toBuilder().apply { | ||||
|                             if (targetTypeAsGenericType != null) { | ||||
|                                 defaultValue("null") | ||||
|                             } | ||||
|                         }.build() | ||||
|                     ) | ||||
|                     addModifiers(KModifier.INLINE) | ||||
|                     targetTypeAsGenericType ?.let { | ||||
|                         addTypeVariable(it) | ||||
|                         returns(it.copy(nullable = nullable)) | ||||
|                     } ?: returns(targetType) | ||||
|                     addCode( | ||||
|                         "return " + (if (nullable) { | ||||
|                             "getOrNull" | ||||
|                         } else { | ||||
|                             "get" | ||||
|                         }) + "(named(\"${name}\"), parameters)" | ||||
|                     ) | ||||
|                 }.build() | ||||
|             ) | ||||
|         } | ||||
|  | ||||
|         addGetterMethod(Scope::class) | ||||
|         addGetterMethod(Koin::class) | ||||
|  | ||||
|         fun FunSpec.Builder.addDefinitionParameter() { | ||||
|             val definitionModifiers = if (targetTypeAsGenericType == null) { | ||||
|                 arrayOf() | ||||
|             } else { | ||||
|                 arrayOf(KModifier.NOINLINE) | ||||
|             } | ||||
|             addParameter( | ||||
|                 ParameterSpec.builder( | ||||
|                     "definition", | ||||
|                     definitionClassName.parameterizedBy(targetType.copy(nullable = false)), | ||||
|                     *definitionModifiers | ||||
|                 ).build() | ||||
|             ) | ||||
|         } | ||||
|  | ||||
|         if (generateSingle) { | ||||
|             fun FunSpec.Builder.configure( | ||||
|                 useInstead: String? = null | ||||
|             ) { | ||||
|                 addKdoc( | ||||
|                     """ | ||||
|                         Will register [definition] with [org.koin.core.module.Module.single] and key "${name}" | ||||
|                     """.trimIndent() | ||||
|                 ) | ||||
|                 receiver(Module::class) | ||||
|                 addParameter( | ||||
|                     ParameterSpec.builder( | ||||
|                         "createdAtStart", | ||||
|                         Boolean::class | ||||
|                     ).apply { | ||||
|                         defaultValue("false") | ||||
|                     }.build() | ||||
|                 ) | ||||
|                 addDefinitionParameter() | ||||
|                 returns(koinDefinitionClassName.parameterizedBy(targetType.copy(nullable = false))) | ||||
|                 addCode( | ||||
|                     "return single(named(\"${name}\"), createdAtStart = createdAtStart, definition = definition)" | ||||
|                 ) | ||||
|                 targetTypeAsGenericType ?.let { | ||||
|                     addTypeVariable(it) | ||||
|                     addModifiers(KModifier.INLINE) | ||||
|                 } | ||||
|                 if (useInstead != null) { | ||||
|                     addAnnotation( | ||||
|                         AnnotationSpec.builder( | ||||
|                             Deprecated::class | ||||
|                         ).apply { | ||||
|                             addMember( | ||||
|                                 CodeBlock.of( | ||||
|                                     """ | ||||
|                                         "This definition is old style and should not be used anymore. Use $useInstead instead" | ||||
|                                     """.trimIndent() | ||||
|                                 ) | ||||
|                             ) | ||||
|                             addMember(CodeBlock.of("ReplaceWith(\"$useInstead\")")) | ||||
|                         }.build() | ||||
|                     ) | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             val actualSingleName = "single${name.replaceFirstChar { it.uppercase() }}" | ||||
|             if (targetTypeAsGenericType == null) { // classic type | ||||
|                 addFunction( | ||||
|                     FunSpec.builder("${name}Single").apply { configure(actualSingleName) }.build() | ||||
|                 ) | ||||
|             } | ||||
|  | ||||
|             addFunction( | ||||
|                 FunSpec.builder(actualSingleName).apply { configure() }.build() | ||||
|             ) | ||||
|         } | ||||
|  | ||||
|         if (generateFactory) { | ||||
|             fun FunSpec.Builder.configure( | ||||
|                 useInstead: String? = null | ||||
|             ) { | ||||
|                 addKdoc( | ||||
|                     """ | ||||
|                         Will register [definition] with [org.koin.core.module.Module.factory] and key "${name}" | ||||
|                     """.trimIndent() | ||||
|                 ) | ||||
|                 receiver(Module::class) | ||||
|                 addDefinitionParameter() | ||||
|                 returns(koinDefinitionClassName.parameterizedBy(targetType.copy(nullable = false))) | ||||
|                 addCode( | ||||
|                     "return factory(named(\"${name}\"), definition = definition)" | ||||
|                 ) | ||||
|                 targetTypeAsGenericType ?.let { | ||||
|                     addTypeVariable(it) | ||||
|                     addModifiers(KModifier.INLINE) | ||||
|                 } | ||||
|                 if (useInstead != null) { | ||||
|                     addAnnotation( | ||||
|                         AnnotationSpec.builder( | ||||
|                             Deprecated::class | ||||
|                         ).apply { | ||||
|                             addMember( | ||||
|                                 CodeBlock.of( | ||||
|                                     """ | ||||
|                                         "This definition is old style and should not be used anymore. Use $useInstead instead" | ||||
|                                     """.trimIndent() | ||||
|                                 ) | ||||
|                             ) | ||||
|                             addMember(CodeBlock.of("ReplaceWith(\"$useInstead\")")) | ||||
|                         }.build() | ||||
|                     ) | ||||
|                 } | ||||
|             } | ||||
|             val actualFactoryName = "factory${name.replaceFirstChar { it.uppercase() }}" | ||||
|             if (targetTypeAsGenericType == null) { // classic type | ||||
|                 addFunction( | ||||
|                     FunSpec.builder("${name}Factory").apply { configure(useInstead = actualFactoryName) }.build() | ||||
|                 ) | ||||
|             } | ||||
|             addFunction( | ||||
|                 FunSpec.builder(actualFactoryName).apply { configure() }.build() | ||||
|             ) | ||||
|         } | ||||
|         addImport("org.koin.core.qualifier", "named") | ||||
|     } | ||||
|  | ||||
|     @OptIn(KspExperimental::class) | ||||
|     override fun process(resolver: Resolver): List<KSAnnotated> { | ||||
|         resolver.getSymbolsWithAnnotation( | ||||
|         (resolver.getSymbolsWithAnnotation( | ||||
|             GenerateKoinDefinition::class.qualifiedName!! | ||||
|         ).filterIsInstance<KSFile>().forEach { ksFile -> | ||||
|         ) + resolver.getSymbolsWithAnnotation( | ||||
|             GenerateGenericKoinDefinition::class.qualifiedName!! | ||||
|         )).filterIsInstance<KSFile>().forEach { ksFile -> | ||||
|             FileSpec.builder( | ||||
|                 ksFile.packageName.asString(), | ||||
|                 "GeneratedDefinitions${ksFile.fileName.removeSuffix(".kt")}" | ||||
| @@ -72,92 +306,12 @@ class Processor( | ||||
|                     }.copy( | ||||
|                         nullable = it.nullable | ||||
|                     ) | ||||
|                     fun addGetterProperty( | ||||
|                         receiver: KClass<*> | ||||
|                     ) { | ||||
|                         addProperty( | ||||
|                             PropertySpec.builder( | ||||
|                                 it.name, | ||||
|                                 targetType, | ||||
|                             ).apply { | ||||
|                                 addKdoc( | ||||
|                                     """ | ||||
|                                         @return Definition by key "${it.name}" | ||||
|                                     """.trimIndent() | ||||
|                                 ) | ||||
|                                 getter( | ||||
|                                     FunSpec.getterBuilder().apply { | ||||
|                                         addCode( | ||||
|                                             "return " + (if (it.nullable) { | ||||
|                                                 "getOrNull" | ||||
|                                             } else { | ||||
|                                                 "get" | ||||
|                                             }) + "(named(\"${it.name}\"))" | ||||
|                                         ) | ||||
|                                     }.build() | ||||
|                                 ) | ||||
|                                 receiver(receiver) | ||||
|                             }.build() | ||||
|                         ) | ||||
|                     } | ||||
|  | ||||
|                     addGetterProperty(Scope::class) | ||||
|                     addGetterProperty(Koin::class) | ||||
|  | ||||
|                     if (it.generateSingle) { | ||||
|                         addFunction( | ||||
|                             FunSpec.builder("${it.name}Single").apply { | ||||
|                                 addKdoc( | ||||
|                                     """ | ||||
|                                         Will register [definition] with [org.koin.core.module.Module.single] and key "${it.name}" | ||||
|                                     """.trimIndent() | ||||
|                                 ) | ||||
|                                 receiver(Module::class) | ||||
|                                 addParameter( | ||||
|                                     ParameterSpec.builder( | ||||
|                                         "createdAtStart", | ||||
|                                         Boolean::class | ||||
|                                     ).apply { | ||||
|                                         defaultValue("false") | ||||
|                                     }.build() | ||||
|                                 ) | ||||
|                                 addParameter( | ||||
|                                     ParameterSpec.builder( | ||||
|                                         "definition", | ||||
|                                         definitionClassName.parameterizedBy(targetType.copy(nullable = false)) | ||||
|                                     ).build() | ||||
|                                 ) | ||||
|                                 returns(koinDefinitionClassName.parameterizedBy(targetType.copy(nullable = false))) | ||||
|                                 addCode( | ||||
|                                     "return single(named(\"${it.name}\"), createdAtStart = createdAtStart, definition = definition)" | ||||
|                                 ) | ||||
|                             }.build() | ||||
|                         ) | ||||
|                     } | ||||
|  | ||||
|                     if (it.generateFactory) { | ||||
|                         addFunction( | ||||
|                             FunSpec.builder("${it.name}Factory").apply { | ||||
|                                 addKdoc( | ||||
|                                     """ | ||||
|                                         Will register [definition] with [org.koin.core.module.Module.factory] and key "${it.name}" | ||||
|                                     """.trimIndent() | ||||
|                                 ) | ||||
|                                 receiver(Module::class) | ||||
|                                 addParameter( | ||||
|                                     ParameterSpec.builder( | ||||
|                                         "definition", | ||||
|                                         definitionClassName.parameterizedBy(targetType.copy(nullable = false)) | ||||
|                                     ).build() | ||||
|                                 ) | ||||
|                                 returns(koinDefinitionClassName.parameterizedBy(targetType.copy(nullable = false))) | ||||
|                                 addCode( | ||||
|                                     "return factory(named(\"${it.name}\"), definition = definition)" | ||||
|                                 ) | ||||
|                             }.build() | ||||
|                         ) | ||||
|                     } | ||||
|                     addImport("org.koin.core.qualifier", "named") | ||||
|                     addCodeForType(targetType, it.name, it.nullable, it.generateSingle, it.generateFactory) | ||||
|                 } | ||||
|                 ksFile.getAnnotationsByType(GenerateGenericKoinDefinition::class).forEach { | ||||
|                     val targetType = TypeVariableName("T", Any::class) | ||||
|                     addCodeForType(targetType, it.name, it.nullable, it.generateSingle, it.generateFactory) | ||||
|                 } | ||||
|             }.build().let { | ||||
|                 File( | ||||
|   | ||||
| @@ -3,12 +3,15 @@ | ||||
| // ORIGINAL FILE: Test.kt | ||||
| package dev.inmo.micro_utils.koin.generator.test | ||||
|  | ||||
| import kotlin.Any | ||||
| import kotlin.Boolean | ||||
| import kotlin.Deprecated | ||||
| import kotlin.String | ||||
| import org.koin.core.Koin | ||||
| import org.koin.core.definition.Definition | ||||
| import org.koin.core.definition.KoinDefinition | ||||
| import org.koin.core.module.Module | ||||
| import org.koin.core.parameter.ParametersDefinition | ||||
| import org.koin.core.qualifier.named | ||||
| import org.koin.core.scope.Scope | ||||
|  | ||||
| @@ -24,15 +27,98 @@ public val Scope.sampleInfo: Test<String> | ||||
| public val Koin.sampleInfo: Test<String> | ||||
|   get() = get(named("sampleInfo")) | ||||
|  | ||||
| /** | ||||
|  * @return Definition by key "sampleInfo" with [parameters] | ||||
|  */ | ||||
| public inline fun Scope.sampleInfo(noinline parameters: ParametersDefinition): Test<String> = | ||||
|     get(named("sampleInfo"), parameters) | ||||
|  | ||||
| /** | ||||
|  * @return Definition by key "sampleInfo" with [parameters] | ||||
|  */ | ||||
| public inline fun Koin.sampleInfo(noinline parameters: ParametersDefinition): Test<String> = | ||||
|     get(named("sampleInfo"), parameters) | ||||
|  | ||||
| /** | ||||
|  * Will register [definition] with [org.koin.core.module.Module.single] and key "sampleInfo" | ||||
|  */ | ||||
| @Deprecated( | ||||
|   "This definition is old style and should not be used anymore. Use singleSampleInfo instead", | ||||
|   ReplaceWith("singleSampleInfo"), | ||||
| ) | ||||
| public fun Module.sampleInfoSingle(createdAtStart: Boolean = false, | ||||
|     definition: Definition<Test<String>>): KoinDefinition<Test<String>> = | ||||
|     single(named("sampleInfo"), createdAtStart = createdAtStart, definition = definition) | ||||
|  | ||||
| /** | ||||
|  * Will register [definition] with [org.koin.core.module.Module.single] and key "sampleInfo" | ||||
|  */ | ||||
| public fun Module.singleSampleInfo(createdAtStart: Boolean = false, | ||||
|     definition: Definition<Test<String>>): KoinDefinition<Test<String>> = | ||||
|     single(named("sampleInfo"), createdAtStart = createdAtStart, definition = definition) | ||||
|  | ||||
| /** | ||||
|  * Will register [definition] with [org.koin.core.module.Module.factory] and key "sampleInfo" | ||||
|  */ | ||||
| @Deprecated( | ||||
|   "This definition is old style and should not be used anymore. Use factorySampleInfo instead", | ||||
|   ReplaceWith("factorySampleInfo"), | ||||
| ) | ||||
| public fun Module.sampleInfoFactory(definition: Definition<Test<String>>): | ||||
|     KoinDefinition<Test<String>> = factory(named("sampleInfo"), definition = definition) | ||||
|  | ||||
| /** | ||||
|  * Will register [definition] with [org.koin.core.module.Module.factory] and key "sampleInfo" | ||||
|  */ | ||||
| public fun Module.factorySampleInfo(definition: Definition<Test<String>>): | ||||
|     KoinDefinition<Test<String>> = factory(named("sampleInfo"), definition = definition) | ||||
|  | ||||
| /** | ||||
|  * @return Definition by key "test" with [parameters] | ||||
|  */ | ||||
| public inline fun <reified T : Any> Scope.test(noinline parameters: ParametersDefinition? = null): T | ||||
|     = get(named("test"), parameters) | ||||
|  | ||||
| /** | ||||
|  * @return Definition by key "test" with [parameters] | ||||
|  */ | ||||
| public inline fun <reified T : Any> Koin.test(noinline parameters: ParametersDefinition? = null): T | ||||
|     = get(named("test"), parameters) | ||||
|  | ||||
| /** | ||||
|  * Will register [definition] with [org.koin.core.module.Module.single] and key "test" | ||||
|  */ | ||||
| public inline fun <reified T : Any> Module.singleTest(createdAtStart: Boolean = false, noinline | ||||
|     definition: Definition<T>): KoinDefinition<T> = single(named("test"), createdAtStart = | ||||
|     createdAtStart, definition = definition) | ||||
|  | ||||
| /** | ||||
|  * Will register [definition] with [org.koin.core.module.Module.factory] and key "test" | ||||
|  */ | ||||
| public inline fun <reified T : Any> Module.factoryTest(noinline definition: Definition<T>): | ||||
|     KoinDefinition<T> = factory(named("test"), definition = definition) | ||||
|  | ||||
| /** | ||||
|  * @return Definition by key "testNullable" with [parameters] | ||||
|  */ | ||||
| public inline fun <reified T : Any> Scope.testNullable(noinline parameters: ParametersDefinition? = | ||||
|     null): T? = getOrNull(named("testNullable"), parameters) | ||||
|  | ||||
| /** | ||||
|  * @return Definition by key "testNullable" with [parameters] | ||||
|  */ | ||||
| public inline fun <reified T : Any> Koin.testNullable(noinline parameters: ParametersDefinition? = | ||||
|     null): T? = getOrNull(named("testNullable"), parameters) | ||||
|  | ||||
| /** | ||||
|  * Will register [definition] with [org.koin.core.module.Module.single] and key "testNullable" | ||||
|  */ | ||||
| public inline fun <reified T : Any> Module.singleTestNullable(createdAtStart: Boolean = false, | ||||
|     noinline definition: Definition<T>): KoinDefinition<T> = single(named("testNullable"), | ||||
|     createdAtStart = createdAtStart, definition = definition) | ||||
|  | ||||
| /** | ||||
|  * Will register [definition] with [org.koin.core.module.Module.factory] and key "testNullable" | ||||
|  */ | ||||
| public inline fun <reified T : Any> Module.factoryTestNullable(noinline definition: Definition<T>): | ||||
|     KoinDefinition<T> = factory(named("testNullable"), definition = definition) | ||||
|   | ||||
| @@ -1,6 +1,9 @@ | ||||
| @file:GenerateKoinDefinition("sampleInfo", Test::class, String::class, nullable = false) | ||||
| @file:GenerateGenericKoinDefinition("test", nullable = false) | ||||
| @file:GenerateGenericKoinDefinition("testNullable", nullable = true) | ||||
| package dev.inmo.micro_utils.koin.generator.test | ||||
|  | ||||
| import dev.inmo.micro_utils.koin.annotations.GenerateGenericKoinDefinition | ||||
| import dev.inmo.micro_utils.koin.annotations.GenerateKoinDefinition | ||||
| import org.koin.core.Koin | ||||
|  | ||||
|   | ||||
							
								
								
									
										40
									
								
								koin/src/commonMain/kotlin/GetWithDefinition.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								koin/src/commonMain/kotlin/GetWithDefinition.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,40 @@ | ||||
| package dev.inmo.micro_utils.koin | ||||
|  | ||||
| import org.koin.core.Koin | ||||
| import org.koin.core.definition.BeanDefinition | ||||
| import org.koin.core.definition.KoinDefinition | ||||
| import org.koin.core.instance.InstanceFactory | ||||
| import org.koin.core.parameter.ParametersDefinition | ||||
| import org.koin.core.scope.Scope | ||||
|  | ||||
| fun <T> Koin.get(definition: BeanDefinition<T>, parameters: ParametersDefinition? = null): T = get( | ||||
|     definition.primaryType, | ||||
|     definition.qualifier, | ||||
|     parameters | ||||
| ) | ||||
|  | ||||
| fun <T> Koin.get(definition: InstanceFactory<T>, parameters: ParametersDefinition? = null): T = get( | ||||
|     definition.beanDefinition, | ||||
|     parameters | ||||
| ) | ||||
|  | ||||
| fun <T> Koin.get(definition: KoinDefinition<T>, parameters: ParametersDefinition? = null): T = get( | ||||
|     definition.factory, | ||||
|     parameters | ||||
| ) | ||||
|  | ||||
| fun <T> Scope.get(definition: BeanDefinition<T>, parameters: ParametersDefinition? = null): T = get( | ||||
|     definition.primaryType, | ||||
|     definition.qualifier, | ||||
|     parameters | ||||
| ) | ||||
|  | ||||
| fun <T> Scope.get(definition: InstanceFactory<T>, parameters: ParametersDefinition? = null): T = get( | ||||
|     definition.beanDefinition, | ||||
|     parameters | ||||
| ) | ||||
|  | ||||
| fun <T> Scope.get(definition: KoinDefinition<T>, parameters: ParametersDefinition? = null): T = get( | ||||
|     definition.factory, | ||||
|     parameters | ||||
| ) | ||||
| @@ -0,0 +1,26 @@ | ||||
| package dev.inmo.micro_utils.koin.annotations | ||||
|  | ||||
| import kotlin.reflect.KClass | ||||
|  | ||||
| /** | ||||
|  * Use this annotation to mark files near to which generator should place generated extensions for koin [org.koin.core.scope.Scope] | ||||
|  * and [org.koin.core.Koin] | ||||
|  * | ||||
|  * @param name Name for definitions. This name will be available as extension for [org.koin.core.scope.Scope] and [org.koin.core.Koin] | ||||
|  * @param type Type of extensions. It is base star-typed class | ||||
|  * @param typeArgs Generic types for [type]. For example, if [type] == `Something::class` and [typeArgs] == `G1::class, | ||||
|  * G2::class`, the result type will be `Something<G1, G2>` | ||||
|  * @param nullable In case when true, extension will not throw error when definition has not been registered in koin | ||||
|  * @param generateSingle Generate definition factory with [org.koin.core.module.Module.single]. You will be able to use | ||||
|  * the extension [org.koin.core.module.Module].[name]Single(createdAtStart/* default false */) { /* your definition */ } | ||||
|  * @param generateFactory Generate definition factory with [org.koin.core.module.Module.factory]. You will be able to use | ||||
|  * the extension [org.koin.core.module.Module].[name]Factory { /* your definition */ } | ||||
|  */ | ||||
| @Target(AnnotationTarget.FILE) | ||||
| @Repeatable | ||||
| annotation class GenerateGenericKoinDefinition( | ||||
|     val name: String, | ||||
|     val nullable: Boolean = true, | ||||
|     val generateSingle: Boolean = true, | ||||
|     val generateFactory: Boolean = true | ||||
| ) | ||||
							
								
								
									
										27
									
								
								koin/src/jvmMain/kotlin/LazyInject.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								koin/src/jvmMain/kotlin/LazyInject.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | ||||
| package dev.inmo.micro_utils.koin | ||||
|  | ||||
| import org.koin.core.parameter.ParametersDefinition | ||||
| import org.koin.core.qualifier.Qualifier | ||||
| import org.koin.java.KoinJavaComponent | ||||
| import kotlin.reflect.KClass | ||||
|  | ||||
| fun <T> lazyInject( | ||||
|     kClassFactory: () -> KClass<*>, | ||||
|     qualifier: Qualifier? = null, | ||||
|     parameters: ParametersDefinition? = null | ||||
| ): Lazy<T> { | ||||
|     return lazy(LazyThreadSafetyMode.SYNCHRONIZED) { | ||||
|         KoinJavaComponent.get(kClassFactory().java, qualifier, parameters) | ||||
|     } | ||||
| } | ||||
|  | ||||
| fun <T> lazyInject( | ||||
|     kClass: KClass<*>, | ||||
|     qualifier: Qualifier? = null, | ||||
|     parameters: ParametersDefinition? = null | ||||
| ): Lazy<T> = lazyInject({ kClass }, qualifier, parameters) | ||||
|  | ||||
| inline fun <reified T> lazyInject( | ||||
|     qualifier: Qualifier? = null, | ||||
|     noinline parameters: ParametersDefinition? = null | ||||
| ): Lazy<T> = lazyInject(T::class, qualifier, parameters) | ||||
| @@ -1,6 +1,6 @@ | ||||
| package dev.inmo.micro_utils.ktor.common | ||||
|  | ||||
| import com.soywiz.klock.DateTime | ||||
| import korlibs.time.DateTime | ||||
|  | ||||
| typealias FromToDateTime = Pair<DateTime?, DateTime?> | ||||
|  | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| package dev.inmo.micro_utils.ktor.server | ||||
|  | ||||
| import com.soywiz.klock.DateTime | ||||
| import korlibs.time.DateTime | ||||
| import dev.inmo.micro_utils.ktor.common.FromToDateTime | ||||
| import io.ktor.http.Parameters | ||||
|  | ||||
|   | ||||
| @@ -14,3 +14,8 @@ dependencies { | ||||
|     api libs.kotlin.poet | ||||
|     api libs.ksp | ||||
| } | ||||
|  | ||||
| java { | ||||
|     sourceCompatibility = JavaVersion.VERSION_1_8 | ||||
|     targetCompatibility = JavaVersion.VERSION_1_8 | ||||
| } | ||||
|   | ||||
| @@ -100,7 +100,11 @@ class Processor( | ||||
|                         primaryConstructor( | ||||
|                             FunSpec.constructorBuilder().apply { | ||||
|                                 ksClassProperties.forEach { | ||||
|                                     addParameter(it.simpleName.getShortName(), it.typeName) | ||||
|                                     addParameter( | ||||
|                                         ParameterSpec.builder(it.simpleName.getShortName(), it.typeName).apply { | ||||
|                                             annotations += it.annotations.map { it.toAnnotationSpec() } | ||||
|                                         }.build() | ||||
|                                     ) | ||||
|                                     typeBuilder.addProperty( | ||||
|                                         PropertySpec.builder(it.simpleName.getShortName(), it.typeName, KModifier.OVERRIDE).apply { | ||||
|                                             initializer(it.simpleName.getShortName()) | ||||
|   | ||||
| @@ -11,18 +11,20 @@ import kotlinx.serialization.Serializable | ||||
| @Serializable | ||||
| @SerialName(value = "NewTest") | ||||
| public data class NewTest( | ||||
|   public override val property1: String, | ||||
|   public override val property2: Int, | ||||
|   public override val parent: ParentTypeId?, | ||||
|   override val property1: String, | ||||
|   override val property2: Int, | ||||
|   @Serializable | ||||
|   override val parent: ParentTypeId?, | ||||
| ) : Test | ||||
|  | ||||
| @Serializable | ||||
| @SerialName(value = "RegisteredTest") | ||||
| public data class RegisteredTest( | ||||
|   public override val id: TestId, | ||||
|   public override val property1: String, | ||||
|   public override val property2: Int, | ||||
|   public override val parent: ParentTypeId?, | ||||
|   override val id: TestId, | ||||
|   override val property1: String, | ||||
|   override val property2: Int, | ||||
|   @Serializable | ||||
|   override val parent: ParentTypeId?, | ||||
| ) : Test, IRegisteredTest | ||||
|  | ||||
| public fun Test.asNew(): NewTest = NewTest(property1, property2, parent) | ||||
|   | ||||
| @@ -17,6 +17,7 @@ typealias ParentTypeId = TestId | ||||
| sealed interface Test { | ||||
|     val property1: String | ||||
|     val property2: Int | ||||
|     @Serializable | ||||
|     val parent: ParentTypeId? | ||||
|  | ||||
|     @GenerateCRUDModelExcludeOverride | ||||
|   | ||||
| @@ -2,6 +2,7 @@ plugins { | ||||
|     id "org.jetbrains.kotlin.multiplatform" | ||||
|     id "org.jetbrains.kotlin.plugin.serialization" | ||||
|     id "application" | ||||
|     id "com.google.devtools.ksp" | ||||
| } | ||||
|  | ||||
| apply from: "$mppJvmJsLinuxMingwProjectPresetPath" | ||||
| @@ -11,6 +12,7 @@ kotlin { | ||||
|         commonMain { | ||||
|             dependencies { | ||||
|                 api internalProject("micro_utils.startup.plugin") | ||||
|                 api internalProject("micro_utils.koin") | ||||
|             } | ||||
|         } | ||||
|         commonTest { | ||||
| @@ -29,3 +31,10 @@ java { | ||||
|     sourceCompatibility = JavaVersion.VERSION_1_8 | ||||
|     targetCompatibility = JavaVersion.VERSION_1_8 | ||||
| } | ||||
|  | ||||
| dependencies { | ||||
|     add("kspCommonMainMetadata", project(":micro_utils.koin.generator")) | ||||
| } | ||||
|  | ||||
| ksp { | ||||
| } | ||||
|   | ||||
| @@ -0,0 +1,38 @@ | ||||
| // THIS CODE HAVE BEEN GENERATED AUTOMATICALLY | ||||
| // TO REGENERATE IT JUST DELETE FILE | ||||
| // ORIGINAL FILE: StartLauncherPlugin.kt | ||||
| package dev.inmo.micro_utils.startup.launcher | ||||
|  | ||||
| import kotlin.Boolean | ||||
| import kotlinx.serialization.json.Json | ||||
| import org.koin.core.Koin | ||||
| import org.koin.core.definition.Definition | ||||
| import org.koin.core.definition.KoinDefinition | ||||
| import org.koin.core.module.Module | ||||
| import org.koin.core.qualifier.named | ||||
| import org.koin.core.scope.Scope | ||||
|  | ||||
| /** | ||||
|  * @return Definition by key "baseJsonProvider" | ||||
|  */ | ||||
| public val Scope.baseJsonProvider: Json? | ||||
|   get() = getOrNull(named("baseJsonProvider")) | ||||
|  | ||||
| /** | ||||
|  * @return Definition by key "baseJsonProvider" | ||||
|  */ | ||||
| public val Koin.baseJsonProvider: Json? | ||||
|   get() = getOrNull(named("baseJsonProvider")) | ||||
|  | ||||
| /** | ||||
|  * Will register [definition] with [org.koin.core.module.Module.single] and key "baseJsonProvider" | ||||
|  */ | ||||
| public fun Module.baseJsonProviderSingle(createdAtStart: Boolean = false, definition: Definition<Json>): | ||||
|     KoinDefinition<Json> = single(named("baseJsonProvider"), createdAtStart = createdAtStart, definition | ||||
|     = definition) | ||||
|  | ||||
| /** | ||||
|  * Will register [definition] with [org.koin.core.module.Module.factory] and key "baseJsonProvider" | ||||
|  */ | ||||
| public fun Module.baseJsonProviderFactory(definition: Definition<Json>): KoinDefinition<Json> = | ||||
|     factory(named("baseJsonProvider"), definition = definition) | ||||
| @@ -1,9 +1,11 @@ | ||||
| @file:GenerateKoinDefinition("baseJsonProvider", Json::class) | ||||
| package dev.inmo.micro_utils.startup.launcher | ||||
|  | ||||
| import dev.inmo.kslog.common.i | ||||
| import dev.inmo.kslog.common.taggedLogger | ||||
| import dev.inmo.kslog.common.w | ||||
| import dev.inmo.micro_utils.coroutines.runCatchingSafely | ||||
| import dev.inmo.micro_utils.koin.annotations.GenerateKoinDefinition | ||||
| import dev.inmo.micro_utils.startup.launcher.StartLauncherPlugin.setupDI | ||||
| import dev.inmo.micro_utils.startup.launcher.StartLauncherPlugin.startPlugin | ||||
| import dev.inmo.micro_utils.startup.plugin.StartPlugin | ||||
| @@ -13,9 +15,10 @@ import kotlinx.coroutines.joinAll | ||||
| import kotlinx.coroutines.launch | ||||
| import kotlinx.serialization.SerialFormat | ||||
| import kotlinx.serialization.StringFormat | ||||
| import kotlinx.serialization.json.Json | ||||
| import kotlinx.serialization.json.JsonObject | ||||
| import kotlinx.serialization.json.decodeFromJsonElement | ||||
| import kotlinx.serialization.json.jsonObject | ||||
| import kotlinx.serialization.modules.SerializersModule | ||||
| import org.koin.core.Koin | ||||
| import org.koin.core.KoinApplication | ||||
| import org.koin.core.context.startKoin | ||||
| @@ -35,7 +38,20 @@ object StartLauncherPlugin : StartPlugin { | ||||
|         single { rawJsonObject } | ||||
|         single { config } | ||||
|         single { CoroutineScope(Dispatchers.Default) } | ||||
|         single { defaultJson } binds arrayOf(StringFormat::class, SerialFormat::class) | ||||
|         single { | ||||
|             val serializersModules = getAll<SerializersModule>().distinct() | ||||
|             val baseJson = baseJsonProvider ?: defaultJson | ||||
|             if (serializersModules.isEmpty()) { | ||||
|                 baseJson | ||||
|             } else { | ||||
|                 Json(baseJson) { | ||||
|                     serializersModule = SerializersModule { | ||||
|                         include(baseJson.serializersModule) | ||||
|                         serializersModules.forEach { include(it) } | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } binds arrayOf(StringFormat::class, SerialFormat::class) | ||||
|  | ||||
|         includes( | ||||
|             config.plugins.mapNotNull { | ||||
|   | ||||
		Reference in New Issue
	
	Block a user