mirror of
				https://github.com/InsanusMokrassar/MicroUtils.git
				synced 2025-10-23 00:00:32 +00:00 
			
		
		
		
	Compare commits
	
		
			32 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 7cc5972ff7 | |||
| 3bbf978b00 | |||
| ed36467600 | |||
| dd0de327fc | |||
| dccd3ce8fd | |||
| fa45e7b696 | |||
| 57f009e8aa | |||
| 04b633a5ea | |||
| 20d42b05bb | |||
| 91ba50f1ff | |||
| f4476c99f9 | |||
| 50f3f586ab | |||
| 36a2d7ec8e | |||
| 4890b5833e | |||
| e20ab89688 | |||
| e557ba8184 | |||
| 8540e21d5a | |||
| 76c04a8506 | |||
| 128632770e | |||
| 31e0800e81 | |||
| 00ca96eec8 | |||
| 077ef2c639 | |||
| e3ea7be0e7 | |||
| 05fd1c2b14 | |||
| affcffe270 | |||
| 62930231e4 | |||
| ad651631ec | |||
| cf1c8f13db | |||
| 9acc69b897 | |||
| 9bc7cbdb50 | |||
| 2ed8443e28 | |||
| 94f598c2b4 | 
							
								
								
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1,4 +1,5 @@ | ||||
| .idea | ||||
| .vscode | ||||
| .kotlin | ||||
| out/* | ||||
| *.iml | ||||
| @@ -9,6 +10,7 @@ settings.xml | ||||
| .gradle/ | ||||
| build/ | ||||
| out/ | ||||
| bin/ | ||||
|  | ||||
| secret.gradle | ||||
| local.properties | ||||
|   | ||||
							
								
								
									
										38
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										38
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -1,5 +1,43 @@ | ||||
| # Changelog | ||||
|  | ||||
| ## 0.26.1 | ||||
|  | ||||
| * `Versions`: | ||||
|   * `Compose`: `1.8.1` -> `1.8.2` | ||||
|   * `Ktor`: `3.2.1` -> `3.2.2` | ||||
| * `Coroutines`: | ||||
|   * Add opportunity to pass logger in subscribe async | ||||
|  | ||||
| ## 0.26.0 | ||||
|  | ||||
| **WARNING!!! SINCE THIS VERSION IF YOU WANT TO USE SOME OF KSP MODULES, SET `ksp.useKSP2=false` IN YOUR `gradle.properties`** (see [gh issue 2491](https://github.com/google/ksp/issues/2491)) | ||||
|  | ||||
| * `Versions`: | ||||
|   * `Kotlin`: `2.1.21` -> `2.2.0` | ||||
|   * `Serialization`: `1.8.1` -> `1.9.0` | ||||
|   * `KSLog`: `1.4.2` -> `1.5.0` | ||||
|   * `Ktor`: `3.1.3` -> `3.2.1` | ||||
|   * `Koin`: `4.0.4` -> `4.1.0` | ||||
|   * `Okio`: `3.12.0` -> `3.15.0` | ||||
|   * `KSP`: `2.1.20-1.0.31` -> `2.2.0-2.0.2` | ||||
|   * `kotlin-poet`: `1.18.1` -> `2.2.0` | ||||
|  | ||||
| ## 0.25.8 | ||||
|  | ||||
| * `Pagination`: | ||||
|   * `Compose`: | ||||
|     * New function `rememberInfinityPagedComponentContext` to create `InfinityPagedComponentContext` | ||||
|     * New variants of `InfinityPagedComponent` component | ||||
|  | ||||
| ## 0.25.7 | ||||
|  | ||||
| * `Versions`: | ||||
|   * `Compose`: `1.8.0` -> `1.8.1` | ||||
|   * `Xerial SQLite`: `3.49.1.0` -> `3.50.1.0` | ||||
|   * `Okio`: `3.11.0` -> `3.12.0` | ||||
|   * `Android AppCompat`: `1.7.0` -> `1.7.1` | ||||
|   * `Android Fragment`: `1.8.6` -> `1.8.8` | ||||
|  | ||||
| ## 0.25.6 | ||||
|  | ||||
| * `Versions`: | ||||
|   | ||||
							
								
								
									
										17
									
								
								build.gradle
									
									
									
									
									
								
							
							
						
						
									
										17
									
								
								build.gradle
									
									
									
									
									
								
							| @@ -19,15 +19,30 @@ buildscript { | ||||
|  | ||||
| plugins { | ||||
|     alias(libs.plugins.versions) | ||||
|     alias(libs.plugins.nmcp.aggregation) | ||||
| } | ||||
|  | ||||
|  | ||||
| if ((project.hasProperty('SONATYPE_USER') || System.getenv('SONATYPE_USER') != null) && (project.hasProperty('SONATYPE_PASSWORD') || System.getenv('SONATYPE_PASSWORD') != null)) { | ||||
|     nmcpAggregation { | ||||
|         centralPortal { | ||||
|             username = project.hasProperty('SONATYPE_USER') ? project.property('SONATYPE_USER') : System.getenv('SONATYPE_USER') | ||||
|             password = project.hasProperty('SONATYPE_PASSWORD') ? project.property('SONATYPE_PASSWORD') : System.getenv('SONATYPE_PASSWORD') | ||||
|             validationTimeout = Duration.ofHours(4) | ||||
|             publishingType = System.getenv('PUBLISHING_TYPE') != "" ? System.getenv('PUBLISHING_TYPE') : "USER_MANAGED" | ||||
|         } | ||||
|  | ||||
|         publishAllProjectsProbablyBreakingProjectIsolation() | ||||
|     } | ||||
| } | ||||
|  | ||||
| allprojects { | ||||
|     repositories { | ||||
|         mavenLocal() | ||||
|         mavenCentral() | ||||
|         google() | ||||
|         maven { url "https://maven.pkg.jetbrains.space/public/p/compose/dev" } | ||||
|         maven { url "https://nexus.inmo.dev/repository/maven-releases/" } | ||||
|         mavenLocal() | ||||
|     } | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| package dev.inmo.micro_utils.coroutines | ||||
|  | ||||
| import dev.inmo.kslog.common.KSLog | ||||
| import kotlinx.coroutines.* | ||||
| import kotlinx.coroutines.channels.Channel | ||||
| import kotlinx.coroutines.flow.consumeAsFlow | ||||
| @@ -7,13 +8,15 @@ import kotlinx.coroutines.flow.consumeAsFlow | ||||
| fun <T> CoroutineScope.actorAsync( | ||||
|     channelCapacity: Int = Channel.UNLIMITED, | ||||
|     markerFactory: suspend (T) -> Any? = { null }, | ||||
|     logger: KSLog = KSLog, | ||||
|     block: suspend (T) -> Unit | ||||
| ): Channel<T> { | ||||
|     val channel = Channel<T>(channelCapacity) | ||||
|     channel.consumeAsFlow().subscribeAsync(this, markerFactory, block) | ||||
|     channel.consumeAsFlow().subscribeAsync(this, markerFactory, logger, block) | ||||
|     return channel | ||||
| } | ||||
|  | ||||
| @Deprecated("Use standard actosAsync instead", ReplaceWith("actorAsync(channelCapacity, markerFactory, block = block)", "dev.inmo.micro_utils.coroutines.actorAsync")) | ||||
| inline fun <T> CoroutineScope.safeActorAsync( | ||||
|     channelCapacity: Int = Channel.UNLIMITED, | ||||
|     noinline onException: ExceptionHandler<Unit> = defaultSafelyExceptionHandler, | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| package dev.inmo.micro_utils.coroutines | ||||
|  | ||||
| import dev.inmo.kslog.common.KSLog | ||||
| import kotlinx.coroutines.* | ||||
| import kotlinx.coroutines.channels.* | ||||
| import kotlinx.coroutines.flow.* | ||||
| @@ -8,6 +9,7 @@ import kotlinx.coroutines.sync.withLock | ||||
|  | ||||
| private class SubscribeAsyncReceiver<T>( | ||||
|     val scope: CoroutineScope, | ||||
|     val logger: KSLog, | ||||
|     output: suspend SubscribeAsyncReceiver<T>.(T) -> Unit | ||||
| ) { | ||||
|     private val dataChannel: Channel<T> = Channel(Channel.UNLIMITED) | ||||
| @@ -15,7 +17,7 @@ private class SubscribeAsyncReceiver<T>( | ||||
|         get() = dataChannel | ||||
|  | ||||
|     init { | ||||
|         scope.launchLoggingDropExceptions { | ||||
|         scope.launchLoggingDropExceptions(logger = logger) { | ||||
|             for (data in dataChannel) { | ||||
|                 output(data) | ||||
|             } | ||||
| @@ -33,13 +35,16 @@ private data class AsyncSubscriptionCommandData<T, M>( | ||||
|     val scope: CoroutineScope, | ||||
|     val markerFactory: suspend (T) -> M, | ||||
|     val block: suspend (T) -> Unit, | ||||
|     val logger: KSLog, | ||||
|     val onEmpty: suspend (M) -> Unit | ||||
| ) : AsyncSubscriptionCommand<T, M> { | ||||
|     override suspend fun invoke(markersMap: MutableMap<M, SubscribeAsyncReceiver<T>>) { | ||||
|         val marker = markerFactory(data) | ||||
|         markersMap.getOrPut(marker) { | ||||
|             SubscribeAsyncReceiver(scope.LinkedSupervisorScope()) { | ||||
|                 safelyWithoutExceptions { block(it) } | ||||
|             SubscribeAsyncReceiver(scope.LinkedSupervisorScope(), logger) { | ||||
|                 runCatchingLogging(logger = logger) { | ||||
|                     block(it) | ||||
|                 } | ||||
|                 if (isEmpty()) { | ||||
|                     onEmpty(marker) | ||||
|                 } | ||||
| @@ -63,6 +68,7 @@ private data class AsyncSubscriptionCommandClearReceiver<T, M>( | ||||
| fun <T, M> Flow<T>.subscribeAsync( | ||||
|     scope: CoroutineScope, | ||||
|     markerFactory: suspend (T) -> M, | ||||
|     logger:  KSLog = KSLog, | ||||
|     block: suspend (T) -> Unit | ||||
| ): Job { | ||||
|     val subscope = scope.LinkedSupervisorScope() | ||||
| @@ -71,8 +77,14 @@ fun <T, M> Flow<T>.subscribeAsync( | ||||
|         it.invoke(markersMap) | ||||
|     } | ||||
|  | ||||
|     val job = subscribeLoggingDropExceptions(subscope) { data -> | ||||
|         val dataCommand = AsyncSubscriptionCommandData(data, subscope, markerFactory, block) { marker -> | ||||
|     val job = subscribeLoggingDropExceptions(subscope, logger = logger) { data -> | ||||
|         val dataCommand = AsyncSubscriptionCommandData( | ||||
|             data = data, | ||||
|             scope = subscope, | ||||
|             markerFactory = markerFactory, | ||||
|             block = block, | ||||
|             logger = logger | ||||
|         ) { marker -> | ||||
|             actor.send( | ||||
|                 AsyncSubscriptionCommandClearReceiver(marker) | ||||
|             ) | ||||
| @@ -85,17 +97,20 @@ fun <T, M> Flow<T>.subscribeAsync( | ||||
|     return job | ||||
| } | ||||
|  | ||||
| @Deprecated("Renamed", ReplaceWith("subscribeLoggingDropExceptionsAsync(scope, markerFactory, block = block)", "dev.inmo.micro_utils.coroutines.subscribeLoggingDropExceptionsAsync")) | ||||
| fun <T, M> Flow<T>.subscribeSafelyAsync( | ||||
|     scope: CoroutineScope, | ||||
|     markerFactory: suspend (T) -> M, | ||||
|     onException: ExceptionHandler<Unit> = defaultSafelyExceptionHandler, | ||||
|     logger:  KSLog = KSLog, | ||||
|     block: suspend (T) -> Unit | ||||
| ) = subscribeAsync(scope, markerFactory) { | ||||
| ) = subscribeAsync(scope, markerFactory, logger) { | ||||
|     safely(onException) { | ||||
|         block(it) | ||||
|     } | ||||
| } | ||||
|  | ||||
| @Deprecated("Renamed", ReplaceWith("subscribeLoggingDropExceptionsAsync(scope, markerFactory, block = block)", "dev.inmo.micro_utils.coroutines.subscribeLoggingDropExceptionsAsync")) | ||||
| fun <T, M> Flow<T>.subscribeSafelyWithoutExceptionsAsync( | ||||
|     scope: CoroutineScope, | ||||
|     markerFactory: suspend (T) -> M, | ||||
| @@ -107,11 +122,22 @@ fun <T, M> Flow<T>.subscribeSafelyWithoutExceptionsAsync( | ||||
|     } | ||||
| } | ||||
|  | ||||
| fun <T, M> Flow<T>.subscribeLoggingDropExceptionsAsync( | ||||
|     scope: CoroutineScope, | ||||
|     markerFactory: suspend (T) -> M, | ||||
|     logger:  KSLog = KSLog, | ||||
|     block: suspend (T) -> Unit | ||||
| ) = subscribeAsync(scope, markerFactory, logger) { | ||||
|     block(it) | ||||
| } | ||||
|  | ||||
| @Deprecated("Renamed", ReplaceWith("subscribeLoggingDropExceptionsAsync(scope, markerFactory, logger, block = block)", "dev.inmo.micro_utils.coroutines.subscribeLoggingDropExceptionsAsync")) | ||||
| fun <T, M> Flow<T>.subscribeSafelySkippingExceptionsAsync( | ||||
|     scope: CoroutineScope, | ||||
|     markerFactory: suspend (T) -> M, | ||||
|     logger:  KSLog = KSLog, | ||||
|     block: suspend (T) -> Unit | ||||
| ) = subscribeAsync(scope, markerFactory) { | ||||
| ) = subscribeAsync(scope, markerFactory, logger) { | ||||
|     safelyWithoutExceptions({ /* do nothing */}) { | ||||
|         block(it) | ||||
|     } | ||||
|   | ||||
| @@ -30,3 +30,13 @@ allprojects { | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| tasks.register("getPublishableModules") { | ||||
|     doLast { | ||||
|         rootProject.subprojects.each { project -> | ||||
|             if (project.plugins.hasPlugin('maven-publish')) { | ||||
|                 println(":${project.name}") | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -8,6 +8,9 @@ android.useAndroidX=true | ||||
| android.enableJetifier=true | ||||
| org.gradle.jvmargs=-Xmx2g -XX:MaxMetaspaceSize=2g | ||||
|  | ||||
| # https://github.com/google/ksp/issues/2491 | ||||
| ksp.useKSP2=false | ||||
|  | ||||
| # JS NPM | ||||
|  | ||||
| crypto_js_version=4.1.1 | ||||
| @@ -15,5 +18,5 @@ crypto_js_version=4.1.1 | ||||
| # Project data | ||||
|  | ||||
| group=dev.inmo | ||||
| version=0.25.6 | ||||
| android_code_version=296 | ||||
| version=0.26.1 | ||||
| android_code_version=300 | ||||
|   | ||||
| @@ -1,42 +1,43 @@ | ||||
| [versions] | ||||
|  | ||||
| kt = "2.1.21" | ||||
| kt-serialization = "1.8.1" | ||||
| kt = "2.2.0" | ||||
| kt-serialization = "1.9.0" | ||||
| kt-coroutines = "1.10.2" | ||||
|  | ||||
| kotlinx-browser = "0.3" | ||||
|  | ||||
| kslog = "1.4.2" | ||||
| kslog = "1.5.0" | ||||
|  | ||||
| jb-compose = "1.8.0" | ||||
| jb-compose = "1.8.2" | ||||
| jb-exposed = "0.61.0" | ||||
| jb-dokka = "2.0.0" | ||||
|  | ||||
| sqlite = "3.49.1.0" | ||||
| sqlite = "3.50.1.0" | ||||
|  | ||||
| korlibs = "5.4.0" | ||||
| uuid = "0.8.4" | ||||
|  | ||||
| ktor = "3.1.3" | ||||
| ktor = "3.2.2" | ||||
|  | ||||
| gh-release = "2.5.2" | ||||
|  | ||||
| koin = "4.0.4" | ||||
| koin = "4.1.0" | ||||
|  | ||||
| okio = "3.11.0" | ||||
| okio = "3.15.0" | ||||
|  | ||||
| ksp = "2.1.20-1.0.31" | ||||
| kotlin-poet = "1.18.1" | ||||
| ksp = "2.2.0-2.0.2" | ||||
| kotlin-poet = "2.2.0" | ||||
|  | ||||
| versions = "0.51.0" | ||||
| versions = "0.52.0" | ||||
| nmcp = "1.0.1" | ||||
|  | ||||
| android-gradle = "8.9.+" | ||||
| dexcount = "4.0.0" | ||||
|  | ||||
| android-coreKtx = "1.16.0" | ||||
| android-recyclerView = "1.4.0" | ||||
| android-appCompat = "1.7.0" | ||||
| android-fragment = "1.8.6" | ||||
| android-appCompat = "1.7.1" | ||||
| android-fragment = "1.8.8" | ||||
| android-espresso = "3.6.1" | ||||
| android-test = "1.2.1" | ||||
|  | ||||
| @@ -123,3 +124,4 @@ jb-compose = { id = "org.jetbrains.compose", version.ref = "jb-compose" } | ||||
| kt-jb-compose = { id = "org.jetbrains.kotlin.plugin.compose", version.ref = "kt" } | ||||
|  | ||||
| versions = { id = "com.github.ben-manes.versions", version.ref = "versions" } | ||||
| nmcp-aggregation = { id = "com.gradleup.nmcp.aggregation", version.ref = "nmcp" } | ||||
|   | ||||
| @@ -1,51 +1,4 @@ | ||||
| import java.nio.charset.StandardCharsets | ||||
| import java.net.http.HttpClient | ||||
| import java.net.http.HttpRequest | ||||
| import java.net.http.HttpResponse | ||||
|  | ||||
| apply plugin: 'maven-publish' | ||||
| // This script work based on https://ossrh-staging-api.central.sonatype.com/swagger-ui/#/default/manual_upload_repository | ||||
| // and getting available open repos and just uploading them | ||||
| if ((project.hasProperty('SONATYPE_USER') || System.getenv('SONATYPE_USER') != null) && (project.hasProperty('SONATYPE_PASSWORD') || System.getenv('SONATYPE_PASSWORD') != null)) { | ||||
|     def taskName = "uploadSonatypePublication" | ||||
|     if (rootProject.tasks.names.contains(taskName) == false) { | ||||
|         rootProject.tasks.register(taskName) { | ||||
|             doLast { | ||||
|                 def username = project.hasProperty('SONATYPE_USER') ? project.property('SONATYPE_USER') : System.getenv('SONATYPE_USER') | ||||
|                 def password = project.hasProperty('SONATYPE_PASSWORD') ? project.property('SONATYPE_PASSWORD') : System.getenv('SONATYPE_PASSWORD') | ||||
|                 def bearer = Base64.getEncoder().encodeToString("$username:$password".getBytes(StandardCharsets.UTF_8)) | ||||
|      | ||||
|                 def client = HttpClient.newHttpClient() | ||||
|                 def request = HttpRequest.newBuilder() | ||||
|                         .uri(URI.create("https://ossrh-staging-api.central.sonatype.com/manual/search/repositories?state=open")) | ||||
|                         .GET() | ||||
|                         .header("Content-Type", "application/json") | ||||
|                         .header("Authorization", "Bearer $bearer") | ||||
|                         .build() | ||||
|      | ||||
|                 def response = client.send(request, HttpResponse.BodyHandlers.ofString()) | ||||
|                 def keys = new ArrayList<String>() | ||||
|                 response.body().findAll("\"key\"[\\s]*:[\\s]*\"[^\"]+\"").forEach { | ||||
|                     def key = it.find("[^\"]+\"\$").find("[^\"]+") | ||||
|                     keys.add(key) | ||||
|                 } | ||||
|                 keys.forEach { | ||||
|                     println("Start uploading $it") | ||||
|                     def uploadRequest = HttpRequest.newBuilder() | ||||
|                             .uri(URI.create("https://ossrh-staging-api.central.sonatype.com/manual/upload/repository/$it?publishing_type=user_managed")) | ||||
|                             .POST(HttpRequest.BodyPublishers.ofString("")) | ||||
|                             .header("Content-Type", "application/json") | ||||
|                             .header("Authorization", "Bearer $bearer") | ||||
|                             .build() | ||||
|                     def uploadResponse = client.send(uploadRequest, HttpResponse.BodyHandlers.ofString()) | ||||
|                     if (uploadResponse.statusCode() != 200) { | ||||
|                         throw IllegalStateException("Faced error of uploading for repo with key $it. Response: $uploadResponse") | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| task javadocsJar(type: Jar) { | ||||
|     archiveClassifier = 'javadoc' | ||||
|   | ||||
| @@ -1 +1 @@ | ||||
| {"licenses":[{"id":"Apache-2.0","title":"Apache Software License 2.0","url":"https://github.com/InsanusMokrassar/MicroUtils/blob/master/LICENSE"}],"mavenConfig":{"name":"${project.name}","description":"It is set of projects with micro tools for avoiding of routines coding","url":"https://github.com/InsanusMokrassar/MicroUtils/","vcsUrl":"https://github.com/InsanusMokrassar/MicroUtils.git","developers":[{"id":"InsanusMokrassar","name":"Aleksei Ovsiannikov","eMail":"ovsyannikov.alexey95@gmail.com"},{"id":"000Sanya","name":"Syrov Aleksandr","eMail":"000sanya.000sanya@gmail.com"}],"repositories":[{"name":"GithubPackages","url":"https://maven.pkg.github.com/InsanusMokrassar/MicroUtils"},{"name":"InmoNexus","url":"https://nexus.inmo.dev/repository/maven-releases/"},{"name":"sonatype","url":"https://ossrh-staging-api.central.sonatype.com/service/local/staging/deploy/maven2/"}],"gpgSigning":{"type":"dev.inmo.kmppscriptbuilder.core.models.GpgSigning.Optional"},"includeCentralSonatypeUploadingScript":true}} | ||||
| {"licenses":[{"id":"Apache-2.0","title":"Apache Software License 2.0","url":"https://github.com/InsanusMokrassar/MicroUtils/blob/master/LICENSE"}],"mavenConfig":{"name":"${project.name}","description":"It is set of projects with micro tools for avoiding of routines coding","url":"https://github.com/InsanusMokrassar/MicroUtils/","vcsUrl":"https://github.com/InsanusMokrassar/MicroUtils.git","developers":[{"id":"InsanusMokrassar","name":"Aleksei Ovsiannikov","eMail":"ovsyannikov.alexey95@gmail.com"},{"id":"000Sanya","name":"Syrov Aleksandr","eMail":"000sanya.000sanya@gmail.com"}],"repositories":[{"name":"GithubPackages","url":"https://maven.pkg.github.com/InsanusMokrassar/MicroUtils"},{"name":"InmoNexus","url":"https://nexus.inmo.dev/repository/maven-releases/"},{"name":"sonatype","url":"https://ossrh-staging-api.central.sonatype.com/service/local/staging/deploy/maven2/"}],"gpgSigning":{"type":"dev.inmo.kmppscriptbuilder.core.models.GpgSigning.Optional"},"includeCentralSonatypeUploadingScript":false}} | ||||
| @@ -1,51 +1,4 @@ | ||||
| import java.nio.charset.StandardCharsets | ||||
| import java.net.http.HttpClient | ||||
| import java.net.http.HttpRequest | ||||
| import java.net.http.HttpResponse | ||||
|  | ||||
| apply plugin: 'maven-publish' | ||||
| // This script work based on https://ossrh-staging-api.central.sonatype.com/swagger-ui/#/default/manual_upload_repository | ||||
| // and getting available open repos and just uploading them | ||||
| if ((project.hasProperty('SONATYPE_USER') || System.getenv('SONATYPE_USER') != null) && (project.hasProperty('SONATYPE_PASSWORD') || System.getenv('SONATYPE_PASSWORD') != null)) { | ||||
|     def taskName = "uploadSonatypePublication" | ||||
|     if (rootProject.tasks.names.contains(taskName) == false) { | ||||
|         rootProject.tasks.register(taskName) { | ||||
|             doLast { | ||||
|                 def username = project.hasProperty('SONATYPE_USER') ? project.property('SONATYPE_USER') : System.getenv('SONATYPE_USER') | ||||
|                 def password = project.hasProperty('SONATYPE_PASSWORD') ? project.property('SONATYPE_PASSWORD') : System.getenv('SONATYPE_PASSWORD') | ||||
|                 def bearer = Base64.getEncoder().encodeToString("$username:$password".getBytes(StandardCharsets.UTF_8)) | ||||
|      | ||||
|                 def client = HttpClient.newHttpClient() | ||||
|                 def request = HttpRequest.newBuilder() | ||||
|                         .uri(URI.create("https://ossrh-staging-api.central.sonatype.com/manual/search/repositories?state=open")) | ||||
|                         .GET() | ||||
|                         .header("Content-Type", "application/json") | ||||
|                         .header("Authorization", "Bearer $bearer") | ||||
|                         .build() | ||||
|      | ||||
|                 def response = client.send(request, HttpResponse.BodyHandlers.ofString()) | ||||
|                 def keys = new ArrayList<String>() | ||||
|                 response.body().findAll("\"key\"[\\s]*:[\\s]*\"[^\"]+\"").forEach { | ||||
|                     def key = it.find("[^\"]+\"\$").find("[^\"]+") | ||||
|                     keys.add(key) | ||||
|                 } | ||||
|                 keys.forEach { | ||||
|                     println("Start uploading $it") | ||||
|                     def uploadRequest = HttpRequest.newBuilder() | ||||
|                             .uri(URI.create("https://ossrh-staging-api.central.sonatype.com/manual/upload/repository/$it?publishing_type=user_managed")) | ||||
|                             .POST(HttpRequest.BodyPublishers.ofString("")) | ||||
|                             .header("Content-Type", "application/json") | ||||
|                             .header("Authorization", "Bearer $bearer") | ||||
|                             .build() | ||||
|                     def uploadResponse = client.send(uploadRequest, HttpResponse.BodyHandlers.ofString()) | ||||
|                     if (uploadResponse.statusCode() != 200) { | ||||
|                         throw IllegalStateException("Faced error of uploading for repo with key $it. Response: $uploadResponse") | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
|  | ||||
| task javadocJar(type: Jar) { | ||||
|   | ||||
| @@ -1 +1 @@ | ||||
| {"licenses":[{"id":"Apache-2.0","title":"Apache Software License 2.0","url":"https://github.com/InsanusMokrassar/MicroUtils/blob/master/LICENSE"}],"mavenConfig":{"name":"${project.name}","description":"It is set of projects with micro tools for avoiding of routines coding","url":"https://github.com/InsanusMokrassar/MicroUtils/","vcsUrl":"https://github.com/InsanusMokrassar/MicroUtils.git","developers":[{"id":"InsanusMokrassar","name":"Aleksei Ovsiannikov","eMail":"ovsyannikov.alexey95@gmail.com"},{"id":"000Sanya","name":"Syrov Aleksandr","eMail":"000sanya.000sanya@gmail.com"}],"repositories":[{"name":"GithubPackages","url":"https://maven.pkg.github.com/InsanusMokrassar/MicroUtils"},{"name":"InmoNexus","url":"https://nexus.inmo.dev/repository/maven-releases/"},{"name":"sonatype","url":"https://ossrh-staging-api.central.sonatype.com/service/local/staging/deploy/maven2/"}],"gpgSigning":{"type":"dev.inmo.kmppscriptbuilder.core.models.GpgSigning.Optional"},"includeCentralSonatypeUploadingScript":true},"type":"JVM"} | ||||
| {"licenses":[{"id":"Apache-2.0","title":"Apache Software License 2.0","url":"https://github.com/InsanusMokrassar/MicroUtils/blob/master/LICENSE"}],"mavenConfig":{"name":"${project.name}","description":"It is set of projects with micro tools for avoiding of routines coding","url":"https://github.com/InsanusMokrassar/MicroUtils/","vcsUrl":"https://github.com/InsanusMokrassar/MicroUtils.git","developers":[{"id":"InsanusMokrassar","name":"Aleksei Ovsiannikov","eMail":"ovsyannikov.alexey95@gmail.com"},{"id":"000Sanya","name":"Syrov Aleksandr","eMail":"000sanya.000sanya@gmail.com"}],"repositories":[{"name":"GithubPackages","url":"https://maven.pkg.github.com/InsanusMokrassar/MicroUtils"},{"name":"InmoNexus","url":"https://nexus.inmo.dev/repository/maven-releases/"},{"name":"sonatype","url":"https://ossrh-staging-api.central.sonatype.com/service/local/staging/deploy/maven2/"}],"gpgSigning":{"type":"dev.inmo.kmppscriptbuilder.core.models.GpgSigning.Optional"},"includeCentralSonatypeUploadingScript":false},"type":"JVM"} | ||||
| @@ -10,6 +10,7 @@ repositories { | ||||
|  | ||||
| dependencies { | ||||
|     api project(":micro_utils.koin") | ||||
|     api project(":micro_utils.ksp.generator") | ||||
|     api libs.kotlin.poet | ||||
|     api libs.ksp | ||||
| } | ||||
|   | ||||
| @@ -26,6 +26,7 @@ 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_ksp.generator.safeClassName | ||||
| import dev.inmo.micro_utils.koin.annotations.GenerateGenericKoinDefinition | ||||
| import dev.inmo.micro_utils.koin.annotations.GenerateKoinDefinition | ||||
| import org.koin.core.Koin | ||||
| @@ -237,15 +238,7 @@ class Processor( | ||||
|                     """.trimIndent() | ||||
|                 ) | ||||
|                 ksFile.getAnnotationsByType(GenerateKoinDefinition::class).forEach { | ||||
|                     val type = runCatching { | ||||
|                         it.type.asTypeName() | ||||
|                     }.getOrElse { e -> | ||||
|                         if (e is KSTypeNotPresentException) { | ||||
|                             e.ksType.toClassName() | ||||
|                         } else { | ||||
|                             throw e | ||||
|                         } | ||||
|                     } | ||||
|                     val type = safeClassName { it.type } | ||||
|                     val targetType = runCatching { | ||||
|                         type.parameterizedBy(*(it.typeArgs.takeIf { it.isNotEmpty() } ?.map { it.asTypeName() } ?.toTypedArray() ?: return@runCatching type)) | ||||
|                     }.getOrElse { e -> | ||||
|   | ||||
| @@ -5,7 +5,6 @@ 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 | ||||
| @@ -30,95 +29,59 @@ public val Koin.sampleInfo: Test<String> | ||||
| /** | ||||
|  * @return Definition by key "sampleInfo" with [parameters] | ||||
|  */ | ||||
| public inline fun Scope.sampleInfo(noinline parameters: ParametersDefinition): Test<String> = | ||||
|     get(named("sampleInfo"), 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) | ||||
| 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) | ||||
| 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) | ||||
| 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) | ||||
| 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) | ||||
| 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) | ||||
| 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) | ||||
| 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) | ||||
| 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) | ||||
| 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) | ||||
| 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) | ||||
| public inline fun <reified T : Any> Module.factoryTestNullable(noinline definition: Definition<T>): KoinDefinition<T> = factory(named("testNullable"), definition = definition) | ||||
|   | ||||
							
								
								
									
										23
									
								
								ksp/generator/src/main/kotlin/KClassTypeName.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								ksp/generator/src/main/kotlin/KClassTypeName.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | ||||
| package dev.inmo.micro_ksp.generator | ||||
|  | ||||
| import com.google.devtools.ksp.KSTypeNotPresentException | ||||
| import com.google.devtools.ksp.KspExperimental | ||||
| import com.squareup.kotlinpoet.ClassName | ||||
| import com.squareup.kotlinpoet.asTypeName | ||||
| import kotlin.reflect.KClass | ||||
|  | ||||
| @Suppress("NOTHING_TO_INLINE") | ||||
| @OptIn(KspExperimental::class) | ||||
| inline fun safeClassName(classnameGetter: () -> KClass<*>) = runCatching { | ||||
|     classnameGetter().asTypeName() | ||||
| }.getOrElse { e -> | ||||
|     if (e is KSTypeNotPresentException) { | ||||
|         ClassName( | ||||
|             e.ksType.declaration.packageName.asString(), | ||||
|             e.ksType.declaration.qualifiedName ?.asString() ?.replaceFirst(e.ksType.declaration.packageName.asString(), "") | ||||
|                 ?: e.ksType.declaration.simpleName.asString() | ||||
|         ) | ||||
|     } else { | ||||
|         throw e | ||||
|     } | ||||
| } | ||||
| @@ -65,6 +65,60 @@ class InfinityPagedComponentContext<T> internal constructor( | ||||
|     } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Creates and remembers an [InfinityPagedComponentContext] for managing infinite pagination in a Compose UI. | ||||
|  * This function is used to create a persistent pagination context that survives recompositions. | ||||
|  * | ||||
|  * @param size Number of items to load per page. | ||||
|  * @param page Initial page number to start pagination from (defaults to 0). | ||||
|  * @param scope [CoroutineScope] to launch pagination operations in. If not provided, a new scope will be created | ||||
|  * using [rememberCoroutineScope]. | ||||
|  * @param loader Suspended function that loads paginated data. Receives the current pagination context and | ||||
|  * pagination parameters, and returns a [PaginationResult] containing the loaded data. | ||||
|  * @return An [InfinityPagedComponentContext] instance that manages the pagination state and operations. | ||||
|  */ | ||||
| @Composable | ||||
| fun <T> rememberInfinityPagedComponentContext( | ||||
|     size: Int, | ||||
|     page: Int = 0, | ||||
|     scope: CoroutineScope = rememberCoroutineScope(), | ||||
|     doReloadInInit: Boolean = true, | ||||
|     loader: suspend InfinityPagedComponentContext<T>.(Pagination) -> PaginationResult<T> | ||||
| ): InfinityPagedComponentContext<T> { | ||||
|     val context = remember { | ||||
|         InfinityPagedComponentContext( | ||||
|             page = page, | ||||
|             size = size, | ||||
|             scope = scope, | ||||
|             loader = loader | ||||
|         ) | ||||
|     } | ||||
|  | ||||
|     LaunchedEffect(context) { | ||||
|         if (doReloadInInit) { | ||||
|             context.reload() | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return context | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Composable function for managing an infinitely paged component. | ||||
|  * | ||||
|  * @param T The type of the paginated data. | ||||
|  * @param block Composable function that renders the UI with the loaded data. When data is in loading state, block will | ||||
|  * receive null as `it` parameter | ||||
|  */ | ||||
| @Composable | ||||
| fun <T> InfinityPagedComponent( | ||||
|     context: InfinityPagedComponentContext<T>, | ||||
|     block: @Composable InfinityPagedComponentContext<T>.(List<T>?) -> Unit | ||||
| ) { | ||||
|     val dataState = context.dataState.collectAsState() | ||||
|     context.block(dataState.value) | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Composable function for managing an infinitely paged component. | ||||
|  * | ||||
| @@ -76,7 +130,7 @@ class InfinityPagedComponentContext<T> internal constructor( | ||||
|  * receive null as `it` parameter | ||||
|  */ | ||||
| @Composable | ||||
| internal fun <T> InfinityPagedComponent( | ||||
| fun <T> InfinityPagedComponent( | ||||
|     page: Int, | ||||
|     size: Int, | ||||
|     loader: suspend InfinityPagedComponentContext<T>.(Pagination) -> PaginationResult<T>, | ||||
| @@ -84,13 +138,8 @@ internal fun <T> InfinityPagedComponent( | ||||
|     block: @Composable InfinityPagedComponentContext<T>.(List<T>?) -> Unit | ||||
| ) { | ||||
|     val scope = predefinedScope ?: rememberCoroutineScope() | ||||
|     val context = remember { InfinityPagedComponentContext<T>(page, size, scope, loader) } | ||||
|     remember { | ||||
|         context.reload() | ||||
|     } | ||||
|  | ||||
|     val dataState = context.dataState.collectAsState() | ||||
|     context.block(dataState.value) | ||||
|     val context = rememberInfinityPagedComponentContext(page = page, size = size, scope = scope, loader = loader) | ||||
|     InfinityPagedComponent(context, block) | ||||
| } | ||||
|  | ||||
| /** | ||||
|   | ||||
		Reference in New Issue
	
	Block a user