mirror of
				https://github.com/InsanusMokrassar/MicroUtils.git
				synced 2025-10-26 17:50:41 +00:00 
			
		
		
		
	Compare commits
	
		
			42 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 7ba6eed453 | |||
| beeb6ecc0a | |||
| 7cdc17a714 | |||
| 4765a950a9 | |||
| 65e8137e08 | |||
| ae546dd9ad | |||
| 8110c42be0 | |||
| bd2b5ae5fc | |||
| 2ddfffa6a9 | |||
| a4b54e861d | |||
| 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 | 
							
								
								
									
										59
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										59
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -1,5 +1,64 @@ | ||||
| # Changelog | ||||
|  | ||||
| ## 0.19.7 | ||||
|  | ||||
| * `Versions`: | ||||
|     * `Coroutines`: `1.7.1` -> `1.7.2` | ||||
|  | ||||
| ## 0.19.6 | ||||
|  | ||||
| * `Versions`: | ||||
|     * `Coroutines`: `1.6.4` -> `1.7.1` | ||||
|     * `Ktor`: `2.3.1` -> `2.3.2` | ||||
|     * `Compose`: `1.4.0` -> `1.4.1` | ||||
|  | ||||
| ## 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`: | ||||
|   | ||||
| @@ -17,6 +17,10 @@ buildscript { | ||||
|     } | ||||
| } | ||||
|  | ||||
| plugins { | ||||
|     alias(libs.plugins.versions) | ||||
| } | ||||
|  | ||||
| allprojects { | ||||
|     repositories { | ||||
|         mavenLocal() | ||||
| @@ -38,3 +42,4 @@ allprojects { | ||||
|  | ||||
| apply from: "./extensions.gradle" | ||||
| apply from: "./github_release.gradle" | ||||
| apply from: "./versions_plugin_setup.gradle" | ||||
|   | ||||
| @@ -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.3 | ||||
| android_code_version=194 | ||||
| version=0.19.7 | ||||
| android_code_version=203 | ||||
|   | ||||
| @@ -2,27 +2,29 @@ | ||||
|  | ||||
| kt = "1.8.20" | ||||
| kt-serialization = "1.5.1" | ||||
| kt-coroutines = "1.6.4" | ||||
| kt-coroutines = "1.7.2" | ||||
|  | ||||
| kslog = "1.1.1" | ||||
|  | ||||
| jb-compose = "1.4.0" | ||||
| jb-compose = "1.4.1" | ||||
| 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.2" | ||||
|  | ||||
| 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.2" | ||||
| kotlin-poet = "1.14.2" | ||||
|  | ||||
| versions = "0.47.0" | ||||
|  | ||||
| android-gradle = "7.4.2" | ||||
| dexcount = "4.0.0" | ||||
| @@ -30,7 +32,7 @@ dexcount = "4.0.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" | ||||
|  | ||||
| @@ -109,3 +111,5 @@ buildscript-android-dexcount = { module = "com.getkeepsafe.dexcount:dexcount-gra | ||||
| [plugins] | ||||
|  | ||||
| jb-compose = { id = "org.jetbrains.compose", version.ref = "jb-compose" } | ||||
|  | ||||
| versions = { id = "com.github.ben-manes.versions", version.ref = "versions" } | ||||
|   | ||||
| @@ -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 | ||||
|   | ||||
							
								
								
									
										11
									
								
								versions_plugin_setup.gradle
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								versions_plugin_setup.gradle
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| def isNonStable = { String version -> | ||||
|     def stableKeyword = ['RELEASE', 'FINAL', 'GA'].any { it -> version.toUpperCase().contains(it) } | ||||
|     def regex = /^[0-9,.v-]+(-r)?$/ | ||||
|     return !stableKeyword && !(version ==~ regex) | ||||
| } | ||||
|  | ||||
| tasks.named("dependencyUpdates").configure { | ||||
|     rejectVersionIf { | ||||
|         isNonStable(it.candidate.version) | ||||
|     } | ||||
| } | ||||
		Reference in New Issue
	
	Block a user