mirror of
				https://github.com/InsanusMokrassar/MicroUtils.git
				synced 2025-10-26 17:50:41 +00:00 
			
		
		
		
	Compare commits
	
		
			36 Commits
		
	
	
		
			v0.25.7
			...
			renovate/g
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | c4ab279e6e | ||
| 9dad353957 | |||
| 89e16b7bdb | |||
| c2965da341 | |||
| ffb072dc5f | |||
| a247dbcb02 | |||
| 1dd71175f4 | |||
| bbe62c0e7b | |||
| 9822ff321b | |||
| b485d485ef | |||
| 0b3d445109 | |||
| d7e48940bc | |||
| 1049eb0fe7 | |||
| c871ef5635 | |||
| 7edfcb20c4 | |||
| 7a1438a2c0 | |||
| 2af8cba8cd | |||
| 27d74c0a62 | |||
| f86d1bfe06 | |||
| 7cc5972ff7 | |||
| 3bbf978b00 | |||
| ed36467600 | |||
| dd0de327fc | |||
| dccd3ce8fd | |||
| fa45e7b696 | |||
| 57f009e8aa | |||
| 04b633a5ea | |||
| 20d42b05bb | |||
| 91ba50f1ff | |||
| f4476c99f9 | |||
| 50f3f586ab | |||
| 36a2d7ec8e | |||
| 4890b5833e | |||
| e20ab89688 | |||
| e557ba8184 | |||
| 8540e21d5a | 
							
								
								
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1,4 +1,5 @@ | |||||||
| .idea | .idea | ||||||
|  | .vscode | ||||||
| .kotlin | .kotlin | ||||||
| out/* | out/* | ||||||
| *.iml | *.iml | ||||||
|   | |||||||
							
								
								
									
										66
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										66
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -1,5 +1,71 @@ | |||||||
| # Changelog | # Changelog | ||||||
|  |  | ||||||
|  | ## 0.26.6 | ||||||
|  |  | ||||||
|  | * `Versions`: | ||||||
|  |     * `Ktor`: `3.3.0` -> `3.3.1` | ||||||
|  |     * `Okio`: `3.16.0` -> `3.16.2` | ||||||
|  |  | ||||||
|  | ## 0.26.5 | ||||||
|  |  | ||||||
|  | * `Versions`: | ||||||
|  |     * `Kotlin`: `2.2.10` -> `2.2.20` | ||||||
|  |     * `KSLog`: `1.5.0` -> `1.5.1` | ||||||
|  |     * `Ktor`: `3.2.3` -> `3.3.0` | ||||||
|  |     * `KotlinX Browser`: `0.3` -> `0.5.0` | ||||||
|  |     * `Koin`: `4.1.0` -> `4.1.1` | ||||||
|  |  | ||||||
|  | ## 0.26.4 | ||||||
|  |  | ||||||
|  | * `Common`: | ||||||
|  |   * Add expect/actual `MPPFilePathSeparator` | ||||||
|  |   * Fix `FileName` realization to take care about system file path separator | ||||||
|  |  | ||||||
|  | ## 0.26.3 | ||||||
|  |  | ||||||
|  | * `Versions`: | ||||||
|  |   * `Kotlin`: `2.2.0` -> `2.2.10` | ||||||
|  |   * `KSP`: `2.2.0-2.0.2` -> `2.2.10-2.0.2` | ||||||
|  |   * `Android CoreKTX`: `1.16.0` -> `1.17.0` | ||||||
|  |   * `Android Fragment`: `1.8.8` -> `1.8.9` | ||||||
|  |  | ||||||
|  | ## 0.26.2 | ||||||
|  |  | ||||||
|  | * `Versions`: | ||||||
|  |   * `Ktor`: `3.2.2` -> `3.2.3` | ||||||
|  |   * `Okio`: `3.15.0` -> `3.16.0` | ||||||
|  | * `Coroutines`: | ||||||
|  |   * Rename `SpecialMutableStateFlow` to `MutableRedeliverStateFlow` | ||||||
|  |  | ||||||
|  | ## 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 | ## 0.25.7 | ||||||
|  |  | ||||||
| * `Versions`: | * `Versions`: | ||||||
|   | |||||||
| @@ -28,8 +28,8 @@ if ((project.hasProperty('SONATYPE_USER') || System.getenv('SONATYPE_USER') != n | |||||||
|         centralPortal { |         centralPortal { | ||||||
|             username = project.hasProperty('SONATYPE_USER') ? project.property('SONATYPE_USER') : System.getenv('SONATYPE_USER') |             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') |             password = project.hasProperty('SONATYPE_PASSWORD') ? project.property('SONATYPE_PASSWORD') : System.getenv('SONATYPE_PASSWORD') | ||||||
|             verificationTimeout = Duration.ofHours(4) |             validationTimeout = Duration.ofHours(4) | ||||||
|             publishingType = "USER_MANAGED" |             publishingType = System.getenv('PUBLISHING_TYPE') != "" ? System.getenv('PUBLISHING_TYPE') : "USER_MANAGED" | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         publishAllProjectsProbablyBreakingProjectIsolation() |         publishAllProjectsProbablyBreakingProjectIsolation() | ||||||
|   | |||||||
| @@ -2,11 +2,9 @@ import androidx.compose.runtime.remember | |||||||
| import androidx.compose.ui.test.ExperimentalTestApi | import androidx.compose.ui.test.ExperimentalTestApi | ||||||
| import androidx.compose.ui.test.runComposeUiTest | import androidx.compose.ui.test.runComposeUiTest | ||||||
| import dev.inmo.micro_utils.common.compose.LoadableComponent | import dev.inmo.micro_utils.common.compose.LoadableComponent | ||||||
| import dev.inmo.micro_utils.coroutines.SpecialMutableStateFlow | import dev.inmo.micro_utils.coroutines.MutableRedeliverStateFlow | ||||||
| import kotlinx.coroutines.flow.MutableSharedFlow |  | ||||||
| import kotlinx.coroutines.flow.filter | import kotlinx.coroutines.flow.filter | ||||||
| import kotlinx.coroutines.flow.first | import kotlinx.coroutines.flow.first | ||||||
| import kotlinx.coroutines.flow.firstOrNull |  | ||||||
| import org.jetbrains.annotations.TestOnly | import org.jetbrains.annotations.TestOnly | ||||||
| import kotlin.test.Test | import kotlin.test.Test | ||||||
| import kotlin.test.assertTrue | import kotlin.test.assertTrue | ||||||
| @@ -16,8 +14,8 @@ class LoadableComponentTests { | |||||||
|     @Test |     @Test | ||||||
|     @TestOnly |     @TestOnly | ||||||
|     fun testSimpleLoad() = runComposeUiTest { |     fun testSimpleLoad() = runComposeUiTest { | ||||||
|         val loadingFlow = SpecialMutableStateFlow<Int>(0) |         val loadingFlow = MutableRedeliverStateFlow<Int>(0) | ||||||
|         val loadedFlow = SpecialMutableStateFlow<Int>(0) |         val loadedFlow = MutableRedeliverStateFlow<Int>(0) | ||||||
|         setContent { |         setContent { | ||||||
|             LoadableComponent<Int>({ |             LoadableComponent<Int>({ | ||||||
|                 loadingFlow.filter { it == 1 }.first() |                 loadingFlow.filter { it == 1 }.first() | ||||||
|   | |||||||
| @@ -7,7 +7,7 @@ import kotlin.jvm.JvmInline | |||||||
| @JvmInline | @JvmInline | ||||||
| value class FileName(val string: String) { | value class FileName(val string: String) { | ||||||
|     val name: String |     val name: String | ||||||
|         get() = withoutSlashAtTheEnd.takeLastWhile { it != '/' } |         get() = withoutSlashAtTheEnd.takeLastWhile { it != MPPFilePathSeparator } | ||||||
|     val extension: String |     val extension: String | ||||||
|         get() = name.takeLastWhile { it != '.' } |         get() = name.takeLastWhile { it != '.' } | ||||||
|     val nameWithoutExtension: String |     val nameWithoutExtension: String | ||||||
| @@ -18,7 +18,7 @@ value class FileName(val string: String) { | |||||||
|             } ?: filename |             } ?: filename | ||||||
|         } |         } | ||||||
|     val withoutSlashAtTheEnd: String |     val withoutSlashAtTheEnd: String | ||||||
|         get() = string.dropLastWhile { it == '/' } |         get() = string.dropLastWhile { it == MPPFilePathSeparator } | ||||||
|     override fun toString(): String = string |     override fun toString(): String = string | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -26,6 +26,7 @@ value class FileName(val string: String) { | |||||||
| expect class MPPFile | expect class MPPFile | ||||||
|  |  | ||||||
| expect val MPPFile.filename: FileName | expect val MPPFile.filename: FileName | ||||||
|  | expect val MPPFilePathSeparator: Char | ||||||
| expect val MPPFile.filesize: Long | expect val MPPFile.filesize: Long | ||||||
| expect val MPPFile.bytesAllocatorSync: ByteArrayAllocator | expect val MPPFile.bytesAllocatorSync: ByteArrayAllocator | ||||||
| expect val MPPFile.bytesAllocator: SuspendByteArrayAllocator | expect val MPPFile.bytesAllocator: SuspendByteArrayAllocator | ||||||
|   | |||||||
| @@ -35,6 +35,10 @@ private suspend fun MPPFile.dirtyReadBytes(): ByteArray = readBytesPromise().awa | |||||||
|  */ |  */ | ||||||
| actual val MPPFile.filename: FileName | actual val MPPFile.filename: FileName | ||||||
|     get() = FileName(name) |     get() = FileName(name) | ||||||
|  |  | ||||||
|  | actual val MPPFilePathSeparator: Char | ||||||
|  |     get() = '/' | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * @suppress |  * @suppress | ||||||
|  */ |  */ | ||||||
|   | |||||||
| @@ -14,6 +14,10 @@ actual typealias MPPFile = File | |||||||
|  */ |  */ | ||||||
| actual val MPPFile.filename: FileName | actual val MPPFile.filename: FileName | ||||||
|     get() = FileName(name) |     get() = FileName(name) | ||||||
|  |  | ||||||
|  | actual val MPPFilePathSeparator: Char | ||||||
|  |     get() = File.separatorChar | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * @suppress |  * @suppress | ||||||
|  */ |  */ | ||||||
|   | |||||||
| @@ -11,6 +11,10 @@ actual typealias MPPFile = Path | |||||||
|  */ |  */ | ||||||
| actual val MPPFile.filename: FileName | actual val MPPFile.filename: FileName | ||||||
|     get() = FileName(toString()) |     get() = FileName(toString()) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | actual val MPPFilePathSeparator: Char = Path.DIRECTORY_SEPARATOR.first() | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * @suppress |  * @suppress | ||||||
|  */ |  */ | ||||||
|   | |||||||
| @@ -37,6 +37,10 @@ private suspend fun MPPFile.dirtyReadBytes(): ByteArray = readBytesPromise().awa | |||||||
|  */ |  */ | ||||||
| actual val MPPFile.filename: FileName | actual val MPPFile.filename: FileName | ||||||
|     get() = FileName(name) |     get() = FileName(name) | ||||||
|  |  | ||||||
|  | actual val MPPFilePathSeparator: Char | ||||||
|  |     get() = '/' | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * @suppress |  * @suppress | ||||||
|  */ |  */ | ||||||
|   | |||||||
| @@ -3,7 +3,7 @@ package dev.inmo.micro_utils.coroutines.compose | |||||||
| import androidx.compose.runtime.Composable | import androidx.compose.runtime.Composable | ||||||
| import androidx.compose.runtime.collectAsState | import androidx.compose.runtime.collectAsState | ||||||
| import androidx.compose.runtime.remember | import androidx.compose.runtime.remember | ||||||
| import dev.inmo.micro_utils.coroutines.SpecialMutableStateFlow | import dev.inmo.micro_utils.coroutines.MutableRedeliverStateFlow | ||||||
| import kotlinx.coroutines.flow.StateFlow | import kotlinx.coroutines.flow.StateFlow | ||||||
| import kotlinx.coroutines.flow.asStateFlow | import kotlinx.coroutines.flow.asStateFlow | ||||||
| import kotlinx.coroutines.flow.debounce | import kotlinx.coroutines.flow.debounce | ||||||
| @@ -16,7 +16,7 @@ import org.jetbrains.compose.web.css.StyleSheet | |||||||
|  * to add `Style(stylesheet)` on every compose function call |  * to add `Style(stylesheet)` on every compose function call | ||||||
|  */ |  */ | ||||||
| object StyleSheetsAggregator { | object StyleSheetsAggregator { | ||||||
|     private val _stylesFlow = SpecialMutableStateFlow<Set<CSSRulesHolder>>(emptySet()) |     private val _stylesFlow = MutableRedeliverStateFlow<Set<CSSRulesHolder>>(emptySet()) | ||||||
|     val stylesFlow: StateFlow<Set<CSSRulesHolder>> = _stylesFlow.asStateFlow() |     val stylesFlow: StateFlow<Set<CSSRulesHolder>> = _stylesFlow.asStateFlow() | ||||||
|  |  | ||||||
|     @Composable |     @Composable | ||||||
|   | |||||||
| @@ -2,7 +2,7 @@ import androidx.compose.material.Button | |||||||
| import androidx.compose.material.Text | import androidx.compose.material.Text | ||||||
| import androidx.compose.runtime.collectAsState | import androidx.compose.runtime.collectAsState | ||||||
| import androidx.compose.ui.test.* | import androidx.compose.ui.test.* | ||||||
| import dev.inmo.micro_utils.coroutines.SpecialMutableStateFlow | import dev.inmo.micro_utils.coroutines.MutableRedeliverStateFlow | ||||||
| import org.jetbrains.annotations.TestOnly | import org.jetbrains.annotations.TestOnly | ||||||
| import kotlin.test.Test | import kotlin.test.Test | ||||||
|  |  | ||||||
| @@ -11,7 +11,7 @@ class FlowStateTests { | |||||||
|     @Test |     @Test | ||||||
|     @TestOnly |     @TestOnly | ||||||
|     fun simpleTest() = runComposeUiTest { |     fun simpleTest() = runComposeUiTest { | ||||||
|         val flowState = SpecialMutableStateFlow(0) |         val flowState = MutableRedeliverStateFlow(0) | ||||||
|         setContent { |         setContent { | ||||||
|             Button({ flowState.value++ }) { Text("Click") } |             Button({ flowState.value++ }) { Text("Click") } | ||||||
|             Text(flowState.collectAsState().value.toString()) |             Text(flowState.collectAsState().value.toString()) | ||||||
|   | |||||||
| @@ -1,5 +1,6 @@ | |||||||
| package dev.inmo.micro_utils.coroutines | package dev.inmo.micro_utils.coroutines | ||||||
|  |  | ||||||
|  | import dev.inmo.kslog.common.KSLog | ||||||
| import kotlinx.coroutines.* | import kotlinx.coroutines.* | ||||||
| import kotlinx.coroutines.channels.Channel | import kotlinx.coroutines.channels.Channel | ||||||
| import kotlinx.coroutines.flow.consumeAsFlow | import kotlinx.coroutines.flow.consumeAsFlow | ||||||
| @@ -7,13 +8,15 @@ import kotlinx.coroutines.flow.consumeAsFlow | |||||||
| fun <T> CoroutineScope.actorAsync( | fun <T> CoroutineScope.actorAsync( | ||||||
|     channelCapacity: Int = Channel.UNLIMITED, |     channelCapacity: Int = Channel.UNLIMITED, | ||||||
|     markerFactory: suspend (T) -> Any? = { null }, |     markerFactory: suspend (T) -> Any? = { null }, | ||||||
|  |     logger: KSLog = KSLog, | ||||||
|     block: suspend (T) -> Unit |     block: suspend (T) -> Unit | ||||||
| ): Channel<T> { | ): Channel<T> { | ||||||
|     val channel = Channel<T>(channelCapacity) |     val channel = Channel<T>(channelCapacity) | ||||||
|     channel.consumeAsFlow().subscribeAsync(this, markerFactory, block) |     channel.consumeAsFlow().subscribeAsync(this, markerFactory, logger, block) | ||||||
|     return channel |     return channel | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @Deprecated("Use standard actosAsync instead", ReplaceWith("actorAsync(channelCapacity, markerFactory, block = block)", "dev.inmo.micro_utils.coroutines.actorAsync")) | ||||||
| inline fun <T> CoroutineScope.safeActorAsync( | inline fun <T> CoroutineScope.safeActorAsync( | ||||||
|     channelCapacity: Int = Channel.UNLIMITED, |     channelCapacity: Int = Channel.UNLIMITED, | ||||||
|     noinline onException: ExceptionHandler<Unit> = defaultSafelyExceptionHandler, |     noinline onException: ExceptionHandler<Unit> = defaultSafelyExceptionHandler, | ||||||
|   | |||||||
| @@ -1,5 +1,6 @@ | |||||||
| package dev.inmo.micro_utils.coroutines | package dev.inmo.micro_utils.coroutines | ||||||
|  |  | ||||||
|  | import dev.inmo.kslog.common.KSLog | ||||||
| import kotlinx.coroutines.* | import kotlinx.coroutines.* | ||||||
| import kotlinx.coroutines.channels.* | import kotlinx.coroutines.channels.* | ||||||
| import kotlinx.coroutines.flow.* | import kotlinx.coroutines.flow.* | ||||||
| @@ -8,6 +9,7 @@ import kotlinx.coroutines.sync.withLock | |||||||
|  |  | ||||||
| private class SubscribeAsyncReceiver<T>( | private class SubscribeAsyncReceiver<T>( | ||||||
|     val scope: CoroutineScope, |     val scope: CoroutineScope, | ||||||
|  |     val logger: KSLog, | ||||||
|     output: suspend SubscribeAsyncReceiver<T>.(T) -> Unit |     output: suspend SubscribeAsyncReceiver<T>.(T) -> Unit | ||||||
| ) { | ) { | ||||||
|     private val dataChannel: Channel<T> = Channel(Channel.UNLIMITED) |     private val dataChannel: Channel<T> = Channel(Channel.UNLIMITED) | ||||||
| @@ -15,7 +17,7 @@ private class SubscribeAsyncReceiver<T>( | |||||||
|         get() = dataChannel |         get() = dataChannel | ||||||
|  |  | ||||||
|     init { |     init { | ||||||
|         scope.launchLoggingDropExceptions { |         scope.launchLoggingDropExceptions(logger = logger) { | ||||||
|             for (data in dataChannel) { |             for (data in dataChannel) { | ||||||
|                 output(data) |                 output(data) | ||||||
|             } |             } | ||||||
| @@ -33,13 +35,16 @@ private data class AsyncSubscriptionCommandData<T, M>( | |||||||
|     val scope: CoroutineScope, |     val scope: CoroutineScope, | ||||||
|     val markerFactory: suspend (T) -> M, |     val markerFactory: suspend (T) -> M, | ||||||
|     val block: suspend (T) -> Unit, |     val block: suspend (T) -> Unit, | ||||||
|  |     val logger: KSLog, | ||||||
|     val onEmpty: suspend (M) -> Unit |     val onEmpty: suspend (M) -> Unit | ||||||
| ) : AsyncSubscriptionCommand<T, M> { | ) : AsyncSubscriptionCommand<T, M> { | ||||||
|     override suspend fun invoke(markersMap: MutableMap<M, SubscribeAsyncReceiver<T>>) { |     override suspend fun invoke(markersMap: MutableMap<M, SubscribeAsyncReceiver<T>>) { | ||||||
|         val marker = markerFactory(data) |         val marker = markerFactory(data) | ||||||
|         markersMap.getOrPut(marker) { |         markersMap.getOrPut(marker) { | ||||||
|             SubscribeAsyncReceiver(scope.LinkedSupervisorScope()) { |             SubscribeAsyncReceiver(scope.LinkedSupervisorScope(), logger) { | ||||||
|                 safelyWithoutExceptions { block(it) } |                 runCatchingLogging(logger = logger) { | ||||||
|  |                     block(it) | ||||||
|  |                 } | ||||||
|                 if (isEmpty()) { |                 if (isEmpty()) { | ||||||
|                     onEmpty(marker) |                     onEmpty(marker) | ||||||
|                 } |                 } | ||||||
| @@ -63,6 +68,7 @@ private data class AsyncSubscriptionCommandClearReceiver<T, M>( | |||||||
| fun <T, M> Flow<T>.subscribeAsync( | fun <T, M> Flow<T>.subscribeAsync( | ||||||
|     scope: CoroutineScope, |     scope: CoroutineScope, | ||||||
|     markerFactory: suspend (T) -> M, |     markerFactory: suspend (T) -> M, | ||||||
|  |     logger:  KSLog = KSLog, | ||||||
|     block: suspend (T) -> Unit |     block: suspend (T) -> Unit | ||||||
| ): Job { | ): Job { | ||||||
|     val subscope = scope.LinkedSupervisorScope() |     val subscope = scope.LinkedSupervisorScope() | ||||||
| @@ -71,8 +77,14 @@ fun <T, M> Flow<T>.subscribeAsync( | |||||||
|         it.invoke(markersMap) |         it.invoke(markersMap) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     val job = subscribeLoggingDropExceptions(subscope) { data -> |     val job = subscribeLoggingDropExceptions(subscope, logger = logger) { data -> | ||||||
|         val dataCommand = AsyncSubscriptionCommandData(data, subscope, markerFactory, block) { marker -> |         val dataCommand = AsyncSubscriptionCommandData( | ||||||
|  |             data = data, | ||||||
|  |             scope = subscope, | ||||||
|  |             markerFactory = markerFactory, | ||||||
|  |             block = block, | ||||||
|  |             logger = logger | ||||||
|  |         ) { marker -> | ||||||
|             actor.send( |             actor.send( | ||||||
|                 AsyncSubscriptionCommandClearReceiver(marker) |                 AsyncSubscriptionCommandClearReceiver(marker) | ||||||
|             ) |             ) | ||||||
| @@ -85,17 +97,20 @@ fun <T, M> Flow<T>.subscribeAsync( | |||||||
|     return job |     return job | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @Deprecated("Renamed", ReplaceWith("subscribeLoggingDropExceptionsAsync(scope, markerFactory, block = block)", "dev.inmo.micro_utils.coroutines.subscribeLoggingDropExceptionsAsync")) | ||||||
| fun <T, M> Flow<T>.subscribeSafelyAsync( | fun <T, M> Flow<T>.subscribeSafelyAsync( | ||||||
|     scope: CoroutineScope, |     scope: CoroutineScope, | ||||||
|     markerFactory: suspend (T) -> M, |     markerFactory: suspend (T) -> M, | ||||||
|     onException: ExceptionHandler<Unit> = defaultSafelyExceptionHandler, |     onException: ExceptionHandler<Unit> = defaultSafelyExceptionHandler, | ||||||
|  |     logger:  KSLog = KSLog, | ||||||
|     block: suspend (T) -> Unit |     block: suspend (T) -> Unit | ||||||
| ) = subscribeAsync(scope, markerFactory) { | ) = subscribeAsync(scope, markerFactory, logger) { | ||||||
|     safely(onException) { |     safely(onException) { | ||||||
|         block(it) |         block(it) | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @Deprecated("Renamed", ReplaceWith("subscribeLoggingDropExceptionsAsync(scope, markerFactory, block = block)", "dev.inmo.micro_utils.coroutines.subscribeLoggingDropExceptionsAsync")) | ||||||
| fun <T, M> Flow<T>.subscribeSafelyWithoutExceptionsAsync( | fun <T, M> Flow<T>.subscribeSafelyWithoutExceptionsAsync( | ||||||
|     scope: CoroutineScope, |     scope: CoroutineScope, | ||||||
|     markerFactory: suspend (T) -> M, |     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( | fun <T, M> Flow<T>.subscribeSafelySkippingExceptionsAsync( | ||||||
|     scope: CoroutineScope, |     scope: CoroutineScope, | ||||||
|     markerFactory: suspend (T) -> M, |     markerFactory: suspend (T) -> M, | ||||||
|  |     logger:  KSLog = KSLog, | ||||||
|     block: suspend (T) -> Unit |     block: suspend (T) -> Unit | ||||||
| ) = subscribeAsync(scope, markerFactory) { | ) = subscribeAsync(scope, markerFactory, logger) { | ||||||
|     safelyWithoutExceptions({ /* do nothing */}) { |     safelyWithoutExceptions({ /* do nothing */}) { | ||||||
|         block(it) |         block(it) | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -1,7 +1,5 @@ | |||||||
| package dev.inmo.micro_utils.coroutines | package dev.inmo.micro_utils.coroutines | ||||||
| 
 | 
 | ||||||
| import kotlinx.coroutines.CoroutineScope |  | ||||||
| import kotlinx.coroutines.Dispatchers |  | ||||||
| import kotlinx.coroutines.ExperimentalCoroutinesApi | import kotlinx.coroutines.ExperimentalCoroutinesApi | ||||||
| import kotlinx.coroutines.InternalCoroutinesApi | import kotlinx.coroutines.InternalCoroutinesApi | ||||||
| import kotlinx.coroutines.channels.BufferOverflow | import kotlinx.coroutines.channels.BufferOverflow | ||||||
| @@ -11,13 +9,12 @@ import kotlinx.coroutines.flow.MutableStateFlow | |||||||
| import kotlinx.coroutines.flow.StateFlow | import kotlinx.coroutines.flow.StateFlow | ||||||
| import kotlinx.coroutines.internal.SynchronizedObject | import kotlinx.coroutines.internal.SynchronizedObject | ||||||
| import kotlinx.coroutines.internal.synchronized | import kotlinx.coroutines.internal.synchronized | ||||||
| import kotlin.coroutines.CoroutineContext |  | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * Works like [StateFlow], but guarantee that latest value update will always be delivered to |  * Works like [StateFlow], but guarantee that latest value update will always be delivered to | ||||||
|  * each active subscriber |  * each active subscriber | ||||||
|  */ |  */ | ||||||
| open class SpecialMutableStateFlow<T>( | open class MutableRedeliverStateFlow<T>( | ||||||
|     initialValue: T |     initialValue: T | ||||||
| ) : MutableStateFlow<T>, FlowCollector<T>, MutableSharedFlow<T> { | ) : MutableStateFlow<T>, FlowCollector<T>, MutableSharedFlow<T> { | ||||||
|     @OptIn(InternalCoroutinesApi::class) |     @OptIn(InternalCoroutinesApi::class) | ||||||
| @@ -68,3 +65,6 @@ open class SpecialMutableStateFlow<T>( | |||||||
| 
 | 
 | ||||||
|     override suspend fun collect(collector: FlowCollector<T>) = sharingFlow.collect(collector) |     override suspend fun collect(collector: FlowCollector<T>) = sharingFlow.collect(collector) | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | @Deprecated("Renamed to MutableRedeliverStateFlow", ReplaceWith("MutableRedeliverStateFlow<T>")) | ||||||
|  | typealias SpecialMutableStateFlow<T> = MutableRedeliverStateFlow<T> | ||||||
| @@ -1,7 +1,6 @@ | |||||||
| package dev.inmo.micro_utils.coroutines | package dev.inmo.micro_utils.coroutines | ||||||
|  |  | ||||||
| import kotlinx.coroutines.currentCoroutineContext | import kotlinx.coroutines.currentCoroutineContext | ||||||
| import kotlinx.coroutines.flow.MutableStateFlow |  | ||||||
| import kotlinx.coroutines.flow.StateFlow | import kotlinx.coroutines.flow.StateFlow | ||||||
| import kotlinx.coroutines.flow.asStateFlow | import kotlinx.coroutines.flow.asStateFlow | ||||||
| import kotlinx.coroutines.flow.first | import kotlinx.coroutines.flow.first | ||||||
| @@ -44,7 +43,7 @@ sealed interface SmartMutex { | |||||||
|      * @param locked Preset state of [isLocked] and its internal [_lockStateFlow] |      * @param locked Preset state of [isLocked] and its internal [_lockStateFlow] | ||||||
|      */ |      */ | ||||||
|     class Mutable(locked: Boolean = false) : SmartMutex { |     class Mutable(locked: Boolean = false) : SmartMutex { | ||||||
|         private val _lockStateFlow = SpecialMutableStateFlow<Boolean>(locked) |         private val _lockStateFlow = MutableRedeliverStateFlow<Boolean>(locked) | ||||||
|         override val lockStateFlow: StateFlow<Boolean> = _lockStateFlow.asStateFlow() |         override val lockStateFlow: StateFlow<Boolean> = _lockStateFlow.asStateFlow() | ||||||
|  |  | ||||||
|         private val internalChangesMutex = Mutex() |         private val internalChangesMutex = Mutex() | ||||||
|   | |||||||
| @@ -1,7 +1,6 @@ | |||||||
| package dev.inmo.micro_utils.coroutines | package dev.inmo.micro_utils.coroutines | ||||||
|  |  | ||||||
| import kotlinx.coroutines.currentCoroutineContext | import kotlinx.coroutines.currentCoroutineContext | ||||||
| import kotlinx.coroutines.flow.MutableStateFlow |  | ||||||
| import kotlinx.coroutines.flow.StateFlow | import kotlinx.coroutines.flow.StateFlow | ||||||
| import kotlinx.coroutines.flow.asStateFlow | import kotlinx.coroutines.flow.asStateFlow | ||||||
| import kotlinx.coroutines.flow.first | import kotlinx.coroutines.flow.first | ||||||
| @@ -47,7 +46,7 @@ sealed interface SmartSemaphore { | |||||||
|      */ |      */ | ||||||
|     class Mutable(permits: Int, acquiredPermits: Int = 0) : SmartSemaphore { |     class Mutable(permits: Int, acquiredPermits: Int = 0) : SmartSemaphore { | ||||||
|         override val maxPermits: Int = permits |         override val maxPermits: Int = permits | ||||||
|         private val _freePermitsStateFlow = SpecialMutableStateFlow<Int>(permits - acquiredPermits) |         private val _freePermitsStateFlow = MutableRedeliverStateFlow<Int>(permits - acquiredPermits) | ||||||
|         override val permitsStateFlow: StateFlow<Int> = _freePermitsStateFlow.asStateFlow() |         override val permitsStateFlow: StateFlow<Int> = _freePermitsStateFlow.asStateFlow() | ||||||
|  |  | ||||||
|         private val internalChangesMutex = Mutex(false) |         private val internalChangesMutex = Mutex(false) | ||||||
|   | |||||||
| @@ -1,33 +1,31 @@ | |||||||
| import dev.inmo.micro_utils.coroutines.SpecialMutableStateFlow | import dev.inmo.micro_utils.coroutines.MutableRedeliverStateFlow | ||||||
| import dev.inmo.micro_utils.coroutines.asDeferred |  | ||||||
| import dev.inmo.micro_utils.coroutines.subscribe | import dev.inmo.micro_utils.coroutines.subscribe | ||||||
| import kotlinx.coroutines.Job | import kotlinx.coroutines.Job | ||||||
| import kotlinx.coroutines.flow.first | import kotlinx.coroutines.flow.first | ||||||
| import kotlinx.coroutines.test.runTest | import kotlinx.coroutines.test.runTest | ||||||
| import kotlin.test.Test | import kotlin.test.Test | ||||||
| import kotlin.test.assertEquals | import kotlin.test.assertEquals | ||||||
| import kotlin.test.assertTrue |  | ||||||
|  |  | ||||||
| class SpecialMutableStateFlowTests { | class SpecialMutableStateFlowTests { | ||||||
|     @Test |     @Test | ||||||
|     fun simpleTest() = runTest { |     fun simpleTest() = runTest { | ||||||
|         val specialMutableStateFlow = SpecialMutableStateFlow(0) |         val mutableRedeliverStateFlow = MutableRedeliverStateFlow(0) | ||||||
|         specialMutableStateFlow.value = 1 |         mutableRedeliverStateFlow.value = 1 | ||||||
|         specialMutableStateFlow.first { it == 1 } |         mutableRedeliverStateFlow.first { it == 1 } | ||||||
|         assertEquals(1, specialMutableStateFlow.value) |         assertEquals(1, mutableRedeliverStateFlow.value) | ||||||
|     } |     } | ||||||
|     @Test |     @Test | ||||||
|     fun specialTest() = runTest { |     fun specialTest() = runTest { | ||||||
|         val specialMutableStateFlow = SpecialMutableStateFlow(0) |         val mutableRedeliverStateFlow = MutableRedeliverStateFlow(0) | ||||||
|         lateinit var subscriberJob: Job |         lateinit var subscriberJob: Job | ||||||
|         subscriberJob = specialMutableStateFlow.subscribe(this) { |         subscriberJob = mutableRedeliverStateFlow.subscribe(this) { | ||||||
|             when (it) { |             when (it) { | ||||||
|                 1 -> specialMutableStateFlow.value = 2 |                 1 -> mutableRedeliverStateFlow.value = 2 | ||||||
|                 2 -> subscriberJob.cancel() |                 2 -> subscriberJob.cancel() | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         specialMutableStateFlow.value = 1 |         mutableRedeliverStateFlow.value = 1 | ||||||
|         subscriberJob.join() |         subscriberJob.join() | ||||||
|         assertEquals(2, specialMutableStateFlow.value) |         assertEquals(2, mutableRedeliverStateFlow.value) | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -8,6 +8,9 @@ android.useAndroidX=true | |||||||
| android.enableJetifier=true | android.enableJetifier=true | ||||||
| org.gradle.jvmargs=-Xmx2g -XX:MaxMetaspaceSize=2g | org.gradle.jvmargs=-Xmx2g -XX:MaxMetaspaceSize=2g | ||||||
|  |  | ||||||
|  | # https://github.com/google/ksp/issues/2491 | ||||||
|  | ksp.useKSP2=false | ||||||
|  |  | ||||||
| # JS NPM | # JS NPM | ||||||
|  |  | ||||||
| crypto_js_version=4.1.1 | crypto_js_version=4.1.1 | ||||||
| @@ -15,5 +18,5 @@ crypto_js_version=4.1.1 | |||||||
| # Project data | # Project data | ||||||
|  |  | ||||||
| group=dev.inmo | group=dev.inmo | ||||||
| version=0.25.7 | version=0.26.6 | ||||||
| android_code_version=297 | android_code_version=305 | ||||||
|   | |||||||
| @@ -1,45 +1,46 @@ | |||||||
| [versions] | [versions] | ||||||
|  |  | ||||||
| kt = "2.1.21" | kt = "2.2.20" | ||||||
| kt-serialization = "1.8.1" | kt-serialization = "1.9.0" | ||||||
| kt-coroutines = "1.10.2" | kt-coroutines = "1.10.2" | ||||||
|  |  | ||||||
| kotlinx-browser = "0.3" | kotlinx-browser = "0.5.0" | ||||||
|  |  | ||||||
| kslog = "1.4.2" | kslog = "1.5.1" | ||||||
|  |  | ||||||
| jb-compose = "1.8.1" | jb-compose = "1.8.2" | ||||||
| jb-exposed = "0.61.0" | jb-exposed = "0.61.0" | ||||||
| jb-dokka = "2.0.0" | jb-dokka = "2.1.0" | ||||||
|  |  | ||||||
|  | # 3.50.3.0 contains bug https://github.com/InsanusMokrassar/MicroUtils/actions/runs/18138301958/job/51629588088 | ||||||
| sqlite = "3.50.1.0" | sqlite = "3.50.1.0" | ||||||
|  |  | ||||||
| korlibs = "5.4.0" | korlibs = "5.4.0" | ||||||
| uuid = "0.8.4" | uuid = "0.8.4" | ||||||
|  |  | ||||||
| ktor = "3.1.3" | ktor = "3.3.1" | ||||||
|  |  | ||||||
| gh-release = "2.5.2" | gh-release = "2.5.2" | ||||||
|  |  | ||||||
| koin = "4.0.4" | koin = "4.1.1" | ||||||
|  |  | ||||||
| okio = "3.12.0" | okio = "3.16.2" | ||||||
|  |  | ||||||
| ksp = "2.1.20-1.0.31" | ksp = "2.2.20-2.0.3" | ||||||
| kotlin-poet = "1.18.1" | kotlin-poet = "2.2.0" | ||||||
|  |  | ||||||
| versions = "0.51.0" | versions = "0.52.0" | ||||||
| nmcp = "0.1.5" | nmcp = "1.1.0" | ||||||
|  |  | ||||||
| android-gradle = "8.9.+" | android-gradle = "8.10.+" | ||||||
| dexcount = "4.0.0" | dexcount = "4.0.0" | ||||||
|  |  | ||||||
| android-coreKtx = "1.16.0" | android-coreKtx = "1.17.0" | ||||||
| android-recyclerView = "1.4.0" | android-recyclerView = "1.4.0" | ||||||
| android-appCompat = "1.7.1" | android-appCompat = "1.7.1" | ||||||
| android-fragment = "1.8.8" | android-fragment = "1.8.9" | ||||||
| android-espresso = "3.6.1" | android-espresso = "3.7.0" | ||||||
| android-test = "1.2.1" | android-test = "1.3.0" | ||||||
|  |  | ||||||
| android-props-minSdk = "21" | android-props-minSdk = "21" | ||||||
| android-props-compileSdk = "36" | android-props-compileSdk = "36" | ||||||
|   | |||||||
							
								
								
									
										
											BIN
										
									
								
								gradle/wrapper/gradle-wrapper.jar
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								gradle/wrapper/gradle-wrapper.jar
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										4
									
								
								gradle/wrapper/gradle-wrapper.properties
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								gradle/wrapper/gradle-wrapper.properties
									
									
									
									
										vendored
									
									
								
							| @@ -1,5 +1,7 @@ | |||||||
| distributionBase=GRADLE_USER_HOME | distributionBase=GRADLE_USER_HOME | ||||||
| distributionPath=wrapper/dists | distributionPath=wrapper/dists | ||||||
| distributionUrl=https\://services.gradle.org/distributions/gradle-8.13-bin.zip | distributionUrl=https\://services.gradle.org/distributions/gradle-9.1.0-bin.zip | ||||||
|  | networkTimeout=10000 | ||||||
|  | validateDistributionUrl=true | ||||||
| zipStoreBase=GRADLE_USER_HOME | zipStoreBase=GRADLE_USER_HOME | ||||||
| zipStorePath=wrapper/dists | zipStorePath=wrapper/dists | ||||||
|   | |||||||
							
								
								
									
										295
									
								
								gradlew
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										295
									
								
								gradlew
									
									
									
									
										vendored
									
									
								
							| @@ -1,7 +1,7 @@ | |||||||
| #!/usr/bin/env sh | #!/bin/sh | ||||||
|  |  | ||||||
| # | # | ||||||
| # Copyright 2015 the original author or authors. | # Copyright © 2015 the original authors. | ||||||
| # | # | ||||||
| # Licensed under the Apache License, Version 2.0 (the "License"); | # Licensed under the Apache License, Version 2.0 (the "License"); | ||||||
| # you may not use this file except in compliance with the License. | # you may not use this file except in compliance with the License. | ||||||
| @@ -15,81 +15,114 @@ | |||||||
| # See the License for the specific language governing permissions and | # See the License for the specific language governing permissions and | ||||||
| # limitations under the License. | # limitations under the License. | ||||||
| # | # | ||||||
|  | # SPDX-License-Identifier: Apache-2.0 | ||||||
|  | # | ||||||
|  |  | ||||||
| ############################################################################## | ############################################################################## | ||||||
| ## | # | ||||||
| ##  Gradle start up script for UN*X | #   Gradle start up script for POSIX generated by Gradle. | ||||||
| ## | # | ||||||
|  | #   Important for running: | ||||||
|  | # | ||||||
|  | #   (1) You need a POSIX-compliant shell to run this script. If your /bin/sh is | ||||||
|  | #       noncompliant, but you have some other compliant shell such as ksh or | ||||||
|  | #       bash, then to run this script, type that shell name before the whole | ||||||
|  | #       command line, like: | ||||||
|  | # | ||||||
|  | #           ksh Gradle | ||||||
|  | # | ||||||
|  | #       Busybox and similar reduced shells will NOT work, because this script | ||||||
|  | #       requires all of these POSIX shell features: | ||||||
|  | #         * functions; | ||||||
|  | #         * expansions «$var», «${var}», «${var:-default}», «${var+SET}», | ||||||
|  | #           «${var#prefix}», «${var%suffix}», and «$( cmd )»; | ||||||
|  | #         * compound commands having a testable exit status, especially «case»; | ||||||
|  | #         * various built-in commands including «command», «set», and «ulimit». | ||||||
|  | # | ||||||
|  | #   Important for patching: | ||||||
|  | # | ||||||
|  | #   (2) This script targets any POSIX shell, so it avoids extensions provided | ||||||
|  | #       by Bash, Ksh, etc; in particular arrays are avoided. | ||||||
|  | # | ||||||
|  | #       The "traditional" practice of packing multiple parameters into a | ||||||
|  | #       space-separated string is a well documented source of bugs and security | ||||||
|  | #       problems, so this is (mostly) avoided, by progressively accumulating | ||||||
|  | #       options in "$@", and eventually passing that to Java. | ||||||
|  | # | ||||||
|  | #       Where the inherited environment variables (DEFAULT_JVM_OPTS, JAVA_OPTS, | ||||||
|  | #       and GRADLE_OPTS) rely on word-splitting, this is performed explicitly; | ||||||
|  | #       see the in-line comments for details. | ||||||
|  | # | ||||||
|  | #       There are tweaks for specific operating systems such as AIX, CygWin, | ||||||
|  | #       Darwin, MinGW, and NonStop. | ||||||
|  | # | ||||||
|  | #   (3) This script is generated from the Groovy template | ||||||
|  | #       https://github.com/gradle/gradle/blob/HEAD/platforms/jvm/plugins-application/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt | ||||||
|  | #       within the Gradle project. | ||||||
|  | # | ||||||
|  | #       You can find Gradle at https://github.com/gradle/gradle/. | ||||||
|  | # | ||||||
| ############################################################################## | ############################################################################## | ||||||
|  |  | ||||||
| # Attempt to set APP_HOME | # Attempt to set APP_HOME | ||||||
|  |  | ||||||
| # Resolve links: $0 may be a link | # Resolve links: $0 may be a link | ||||||
| PRG="$0" | app_path=$0 | ||||||
| # Need this for relative symlinks. |  | ||||||
| while [ -h "$PRG" ] ; do | # Need this for daisy-chained symlinks. | ||||||
|     ls=`ls -ld "$PRG"` | while | ||||||
|     link=`expr "$ls" : '.*-> \(.*\)$'` |     APP_HOME=${app_path%"${app_path##*/}"}  # leaves a trailing /; empty if no leading path | ||||||
|     if expr "$link" : '/.*' > /dev/null; then |     [ -h "$app_path" ] | ||||||
|         PRG="$link" | do | ||||||
|     else |     ls=$( ls -ld "$app_path" ) | ||||||
|         PRG=`dirname "$PRG"`"/$link" |     link=${ls#*' -> '} | ||||||
|     fi |     case $link in             #( | ||||||
|  |       /*)   app_path=$link ;; #( | ||||||
|  |       *)    app_path=$APP_HOME$link ;; | ||||||
|  |     esac | ||||||
| done | done | ||||||
| SAVED="`pwd`" |  | ||||||
| cd "`dirname \"$PRG\"`/" >/dev/null |  | ||||||
| APP_HOME="`pwd -P`" |  | ||||||
| cd "$SAVED" >/dev/null |  | ||||||
|  |  | ||||||
| APP_NAME="Gradle" | # This is normally unused | ||||||
| APP_BASE_NAME=`basename "$0"` | # shellcheck disable=SC2034 | ||||||
|  | APP_BASE_NAME=${0##*/} | ||||||
| # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. | # Discard cd standard output in case $CDPATH is set (https://github.com/gradle/gradle/issues/25036) | ||||||
| DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"' | APP_HOME=$( cd -P "${APP_HOME:-./}" > /dev/null && printf '%s\n' "$PWD" ) || exit | ||||||
|  |  | ||||||
| # Use the maximum available, or set MAX_FD != -1 to use that value. | # Use the maximum available, or set MAX_FD != -1 to use that value. | ||||||
| MAX_FD="maximum" | MAX_FD=maximum | ||||||
|  |  | ||||||
| warn () { | warn () { | ||||||
|     echo "$*" |     echo "$*" | ||||||
| } | } >&2 | ||||||
|  |  | ||||||
| die () { | die () { | ||||||
|     echo |     echo | ||||||
|     echo "$*" |     echo "$*" | ||||||
|     echo |     echo | ||||||
|     exit 1 |     exit 1 | ||||||
| } | } >&2 | ||||||
|  |  | ||||||
| # OS specific support (must be 'true' or 'false'). | # OS specific support (must be 'true' or 'false'). | ||||||
| cygwin=false | cygwin=false | ||||||
| msys=false | msys=false | ||||||
| darwin=false | darwin=false | ||||||
| nonstop=false | nonstop=false | ||||||
| case "`uname`" in | case "$( uname )" in                #( | ||||||
|   CYGWIN* ) |   CYGWIN* )         cygwin=true  ;; #( | ||||||
|     cygwin=true |   Darwin* )         darwin=true  ;; #( | ||||||
|     ;; |   MSYS* | MINGW* )  msys=true    ;; #( | ||||||
|   Darwin* ) |   NONSTOP* )        nonstop=true ;; | ||||||
|     darwin=true |  | ||||||
|     ;; |  | ||||||
|   MINGW* ) |  | ||||||
|     msys=true |  | ||||||
|     ;; |  | ||||||
|   NONSTOP* ) |  | ||||||
|     nonstop=true |  | ||||||
|     ;; |  | ||||||
| esac | esac | ||||||
|  |  | ||||||
| CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar |  | ||||||
|  |  | ||||||
|  |  | ||||||
| # Determine the Java command to use to start the JVM. | # Determine the Java command to use to start the JVM. | ||||||
| if [ -n "$JAVA_HOME" ] ; then | if [ -n "$JAVA_HOME" ] ; then | ||||||
|     if [ -x "$JAVA_HOME/jre/sh/java" ] ; then |     if [ -x "$JAVA_HOME/jre/sh/java" ] ; then | ||||||
|         # IBM's JDK on AIX uses strange locations for the executables |         # IBM's JDK on AIX uses strange locations for the executables | ||||||
|         JAVACMD="$JAVA_HOME/jre/sh/java" |         JAVACMD=$JAVA_HOME/jre/sh/java | ||||||
|     else |     else | ||||||
|         JAVACMD="$JAVA_HOME/bin/java" |         JAVACMD=$JAVA_HOME/bin/java | ||||||
|     fi |     fi | ||||||
|     if [ ! -x "$JAVACMD" ] ; then |     if [ ! -x "$JAVACMD" ] ; then | ||||||
|         die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME |         die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME | ||||||
| @@ -98,88 +131,118 @@ Please set the JAVA_HOME variable in your environment to match the | |||||||
| location of your Java installation." | location of your Java installation." | ||||||
|     fi |     fi | ||||||
| else | else | ||||||
|     JAVACMD="java" |     JAVACMD=java | ||||||
|     which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. |     if ! command -v java >/dev/null 2>&1 | ||||||
|  |     then | ||||||
|  |         die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. | ||||||
|  |  | ||||||
| Please set the JAVA_HOME variable in your environment to match the | Please set the JAVA_HOME variable in your environment to match the | ||||||
| location of your Java installation." | location of your Java installation." | ||||||
|     fi |     fi | ||||||
|  | fi | ||||||
|  |  | ||||||
| # Increase the maximum file descriptors if we can. | # Increase the maximum file descriptors if we can. | ||||||
| if [ "$cygwin" = "false" -a "$darwin" = "false" -a "$nonstop" = "false" ] ; then | if ! "$cygwin" && ! "$darwin" && ! "$nonstop" ; then | ||||||
|     MAX_FD_LIMIT=`ulimit -H -n` |     case $MAX_FD in #( | ||||||
|     if [ $? -eq 0 ] ; then |       max*) | ||||||
|         if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then |         # In POSIX sh, ulimit -H is undefined. That's why the result is checked to see if it worked. | ||||||
|             MAX_FD="$MAX_FD_LIMIT" |         # shellcheck disable=SC2039,SC3045 | ||||||
|         fi |         MAX_FD=$( ulimit -H -n ) || | ||||||
|         ulimit -n $MAX_FD |             warn "Could not query maximum file descriptor limit" | ||||||
|         if [ $? -ne 0 ] ; then |     esac | ||||||
|             warn "Could not set maximum file descriptor limit: $MAX_FD" |     case $MAX_FD in  #( | ||||||
|         fi |       '' | soft) :;; #( | ||||||
|     else |       *) | ||||||
|         warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT" |         # In POSIX sh, ulimit -n is undefined. That's why the result is checked to see if it worked. | ||||||
|     fi |         # shellcheck disable=SC2039,SC3045 | ||||||
| fi |         ulimit -n "$MAX_FD" || | ||||||
|  |             warn "Could not set maximum file descriptor limit to $MAX_FD" | ||||||
| # For Darwin, add options to specify how the application appears in the dock |  | ||||||
| if $darwin; then |  | ||||||
|     GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\"" |  | ||||||
| fi |  | ||||||
|  |  | ||||||
| # For Cygwin or MSYS, switch paths to Windows format before running java |  | ||||||
| if [ "$cygwin" = "true" -o "$msys" = "true" ] ; then |  | ||||||
|     APP_HOME=`cygpath --path --mixed "$APP_HOME"` |  | ||||||
|     CLASSPATH=`cygpath --path --mixed "$CLASSPATH"` |  | ||||||
|      |  | ||||||
|     JAVACMD=`cygpath --unix "$JAVACMD"` |  | ||||||
|  |  | ||||||
|     # We build the pattern for arguments to be converted via cygpath |  | ||||||
|     ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null` |  | ||||||
|     SEP="" |  | ||||||
|     for dir in $ROOTDIRSRAW ; do |  | ||||||
|         ROOTDIRS="$ROOTDIRS$SEP$dir" |  | ||||||
|         SEP="|" |  | ||||||
|     done |  | ||||||
|     OURCYGPATTERN="(^($ROOTDIRS))" |  | ||||||
|     # Add a user-defined pattern to the cygpath arguments |  | ||||||
|     if [ "$GRADLE_CYGPATTERN" != "" ] ; then |  | ||||||
|         OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)" |  | ||||||
|     fi |  | ||||||
|     # Now convert the arguments - kludge to limit ourselves to /bin/sh |  | ||||||
|     i=0 |  | ||||||
|     for arg in "$@" ; do |  | ||||||
|         CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -` |  | ||||||
|         CHECK2=`echo "$arg"|egrep -c "^-"`                                 ### Determine if an option |  | ||||||
|  |  | ||||||
|         if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then                    ### Added a condition |  | ||||||
|             eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"` |  | ||||||
|         else |  | ||||||
|             eval `echo args$i`="\"$arg\"" |  | ||||||
|         fi |  | ||||||
|         i=`expr $i + 1` |  | ||||||
|     done |  | ||||||
|     case $i in |  | ||||||
|         0) set -- ;; |  | ||||||
|         1) set -- "$args0" ;; |  | ||||||
|         2) set -- "$args0" "$args1" ;; |  | ||||||
|         3) set -- "$args0" "$args1" "$args2" ;; |  | ||||||
|         4) set -- "$args0" "$args1" "$args2" "$args3" ;; |  | ||||||
|         5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;; |  | ||||||
|         6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;; |  | ||||||
|         7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;; |  | ||||||
|         8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;; |  | ||||||
|         9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;; |  | ||||||
|     esac |     esac | ||||||
| fi | fi | ||||||
|  |  | ||||||
| # Escape application args | # Collect all arguments for the java command, stacking in reverse order: | ||||||
| save () { | #   * args from the command line | ||||||
|     for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done | #   * the main class name | ||||||
|     echo " " | #   * -classpath | ||||||
| } | #   * -D...appname settings | ||||||
| APP_ARGS=`save "$@"` | #   * --module-path (only if needed) | ||||||
|  | #   * DEFAULT_JVM_OPTS, JAVA_OPTS, and GRADLE_OPTS environment variables. | ||||||
|  |  | ||||||
| # Collect all arguments for the java command, following the shell quoting and substitution rules | # For Cygwin or MSYS, switch paths to Windows format before running java | ||||||
| eval set -- $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS "\"-Dorg.gradle.appname=$APP_BASE_NAME\"" -classpath "\"$CLASSPATH\"" org.gradle.wrapper.GradleWrapperMain "$APP_ARGS" | if "$cygwin" || "$msys" ; then | ||||||
|  |     APP_HOME=$( cygpath --path --mixed "$APP_HOME" ) | ||||||
|  |  | ||||||
|  |     JAVACMD=$( cygpath --unix "$JAVACMD" ) | ||||||
|  |  | ||||||
|  |     # Now convert the arguments - kludge to limit ourselves to /bin/sh | ||||||
|  |     for arg do | ||||||
|  |         if | ||||||
|  |             case $arg in                                #( | ||||||
|  |               -*)   false ;;                            # don't mess with options #( | ||||||
|  |               /?*)  t=${arg#/} t=/${t%%/*}              # looks like a POSIX filepath | ||||||
|  |                     [ -e "$t" ] ;;                      #( | ||||||
|  |               *)    false ;; | ||||||
|  |             esac | ||||||
|  |         then | ||||||
|  |             arg=$( cygpath --path --ignore --mixed "$arg" ) | ||||||
|  |         fi | ||||||
|  |         # Roll the args list around exactly as many times as the number of | ||||||
|  |         # args, so each arg winds up back in the position where it started, but | ||||||
|  |         # possibly modified. | ||||||
|  |         # | ||||||
|  |         # NB: a `for` loop captures its iteration list before it begins, so | ||||||
|  |         # changing the positional parameters here affects neither the number of | ||||||
|  |         # iterations, nor the values presented in `arg`. | ||||||
|  |         shift                   # remove old arg | ||||||
|  |         set -- "$@" "$arg"      # push replacement arg | ||||||
|  |     done | ||||||
|  | fi | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. | ||||||
|  | DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"' | ||||||
|  |  | ||||||
|  | # Collect all arguments for the java command: | ||||||
|  | #   * DEFAULT_JVM_OPTS, JAVA_OPTS, and optsEnvironmentVar are not allowed to contain shell fragments, | ||||||
|  | #     and any embedded shellness will be escaped. | ||||||
|  | #   * For example: A user cannot expect ${Hostname} to be expanded, as it is an environment variable and will be | ||||||
|  | #     treated as '${Hostname}' itself on the command line. | ||||||
|  |  | ||||||
|  | set -- \ | ||||||
|  |         "-Dorg.gradle.appname=$APP_BASE_NAME" \ | ||||||
|  |         -jar "$APP_HOME/gradle/wrapper/gradle-wrapper.jar" \ | ||||||
|  |         "$@" | ||||||
|  |  | ||||||
|  | # Stop when "xargs" is not available. | ||||||
|  | if ! command -v xargs >/dev/null 2>&1 | ||||||
|  | then | ||||||
|  |     die "xargs is not available" | ||||||
|  | fi | ||||||
|  |  | ||||||
|  | # Use "xargs" to parse quoted args. | ||||||
|  | # | ||||||
|  | # With -n1 it outputs one arg per line, with the quotes and backslashes removed. | ||||||
|  | # | ||||||
|  | # In Bash we could simply go: | ||||||
|  | # | ||||||
|  | #   readarray ARGS < <( xargs -n1 <<<"$var" ) && | ||||||
|  | #   set -- "${ARGS[@]}" "$@" | ||||||
|  | # | ||||||
|  | # but POSIX shell has neither arrays nor command substitution, so instead we | ||||||
|  | # post-process each arg (as a line of input to sed) to backslash-escape any | ||||||
|  | # character that might be a shell metacharacter, then use eval to reverse | ||||||
|  | # that process (while maintaining the separation between arguments), and wrap | ||||||
|  | # the whole thing up as a single "set" statement. | ||||||
|  | # | ||||||
|  | # This will of course break if any of these variables contains a newline or | ||||||
|  | # an unmatched quote. | ||||||
|  | # | ||||||
|  |  | ||||||
|  | eval "set -- $( | ||||||
|  |         printf '%s\n' "$DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS" | | ||||||
|  |         xargs -n1 | | ||||||
|  |         sed ' s~[^-[:alnum:]+,./:=@_]~\\&~g; ' | | ||||||
|  |         tr '\n' ' ' | ||||||
|  |     )" '"$@"' | ||||||
|  |  | ||||||
| exec "$JAVACMD" "$@" | exec "$JAVACMD" "$@" | ||||||
|   | |||||||
							
								
								
									
										53
									
								
								gradlew.bat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										53
									
								
								gradlew.bat
									
									
									
									
										vendored
									
									
								
							| @@ -13,6 +13,8 @@ | |||||||
| @rem See the License for the specific language governing permissions and | @rem See the License for the specific language governing permissions and | ||||||
| @rem limitations under the License. | @rem limitations under the License. | ||||||
| @rem | @rem | ||||||
|  | @rem SPDX-License-Identifier: Apache-2.0 | ||||||
|  | @rem | ||||||
|  |  | ||||||
| @if "%DEBUG%"=="" @echo off | @if "%DEBUG%"=="" @echo off | ||||||
| @rem ########################################################################## | @rem ########################################################################## | ||||||
| @@ -26,6 +28,7 @@ if "%OS%"=="Windows_NT" setlocal | |||||||
|  |  | ||||||
| set DIRNAME=%~dp0 | set DIRNAME=%~dp0 | ||||||
| if "%DIRNAME%"=="" set DIRNAME=. | if "%DIRNAME%"=="" set DIRNAME=. | ||||||
|  | @rem This is normally unused | ||||||
| set APP_BASE_NAME=%~n0 | set APP_BASE_NAME=%~n0 | ||||||
| set APP_HOME=%DIRNAME% | set APP_HOME=%DIRNAME% | ||||||
|  |  | ||||||
| @@ -40,13 +43,13 @@ if defined JAVA_HOME goto findJavaFromJavaHome | |||||||
|  |  | ||||||
| set JAVA_EXE=java.exe | set JAVA_EXE=java.exe | ||||||
| %JAVA_EXE% -version >NUL 2>&1 | %JAVA_EXE% -version >NUL 2>&1 | ||||||
| if "%ERRORLEVEL%" == "0" goto init | if %ERRORLEVEL% equ 0 goto execute | ||||||
|  |  | ||||||
| echo. | echo. 1>&2 | ||||||
| echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. | echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 1>&2 | ||||||
| echo. | echo. 1>&2 | ||||||
| echo Please set the JAVA_HOME variable in your environment to match the | echo Please set the JAVA_HOME variable in your environment to match the 1>&2 | ||||||
| echo location of your Java installation. | echo location of your Java installation. 1>&2 | ||||||
|  |  | ||||||
| goto fail | goto fail | ||||||
|  |  | ||||||
| @@ -54,49 +57,35 @@ goto fail | |||||||
| set JAVA_HOME=%JAVA_HOME:"=% | set JAVA_HOME=%JAVA_HOME:"=% | ||||||
| set JAVA_EXE=%JAVA_HOME%/bin/java.exe | set JAVA_EXE=%JAVA_HOME%/bin/java.exe | ||||||
|  |  | ||||||
| if exist "%JAVA_EXE%" goto init | if exist "%JAVA_EXE%" goto execute | ||||||
|  |  | ||||||
| echo. | echo. 1>&2 | ||||||
| echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% | echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% 1>&2 | ||||||
| echo. | echo. 1>&2 | ||||||
| echo Please set the JAVA_HOME variable in your environment to match the | echo Please set the JAVA_HOME variable in your environment to match the 1>&2 | ||||||
| echo location of your Java installation. | echo location of your Java installation. 1>&2 | ||||||
|  |  | ||||||
| goto fail | goto fail | ||||||
|  |  | ||||||
| :init |  | ||||||
| @rem Get command-line arguments, handling Windows variants |  | ||||||
|  |  | ||||||
| if not "%OS%" == "Windows_NT" goto win9xME_args |  | ||||||
|  |  | ||||||
| :win9xME_args |  | ||||||
| @rem Slurp the command line arguments. |  | ||||||
| set CMD_LINE_ARGS= |  | ||||||
| set _SKIP=2 |  | ||||||
|  |  | ||||||
| :win9xME_args_slurp |  | ||||||
| if "x%~1" == "x" goto execute |  | ||||||
|  |  | ||||||
| set CMD_LINE_ARGS=%* |  | ||||||
|  |  | ||||||
| :execute | :execute | ||||||
| @rem Setup the command line | @rem Setup the command line | ||||||
|  |  | ||||||
| set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @rem Execute Gradle | @rem Execute Gradle | ||||||
| "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS% | "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -jar "%APP_HOME%\gradle\wrapper\gradle-wrapper.jar" %* | ||||||
|  |  | ||||||
| :end | :end | ||||||
| @rem End local scope for the variables with windows NT shell | @rem End local scope for the variables with windows NT shell | ||||||
| if "%ERRORLEVEL%"=="0" goto mainEnd | if %ERRORLEVEL% equ 0 goto mainEnd | ||||||
|  |  | ||||||
| :fail | :fail | ||||||
| rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of | rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of | ||||||
| rem the _cmd.exe /c_ return code! | rem the _cmd.exe /c_ return code! | ||||||
| if  not "" == "%GRADLE_EXIT_CONSOLE%" exit 1 | set EXIT_CODE=%ERRORLEVEL% | ||||||
| exit /b 1 | if %EXIT_CODE% equ 0 set EXIT_CODE=1 | ||||||
|  | if not ""=="%GRADLE_EXIT_CONSOLE%" exit %EXIT_CODE% | ||||||
|  | exit /b %EXIT_CODE% | ||||||
|  |  | ||||||
| :mainEnd | :mainEnd | ||||||
| if "%OS%"=="Windows_NT" endlocal | if "%OS%"=="Windows_NT" endlocal | ||||||
|   | |||||||
| @@ -10,6 +10,7 @@ repositories { | |||||||
|  |  | ||||||
| dependencies { | dependencies { | ||||||
|     api project(":micro_utils.koin") |     api project(":micro_utils.koin") | ||||||
|  |     api project(":micro_utils.ksp.generator") | ||||||
|     api libs.kotlin.poet |     api libs.kotlin.poet | ||||||
|     api libs.ksp |     api libs.ksp | ||||||
| } | } | ||||||
|   | |||||||
| @@ -26,6 +26,7 @@ import com.squareup.kotlinpoet.asTypeName | |||||||
| import com.squareup.kotlinpoet.ksp.toClassName | import com.squareup.kotlinpoet.ksp.toClassName | ||||||
| import com.squareup.kotlinpoet.ksp.toTypeName | import com.squareup.kotlinpoet.ksp.toTypeName | ||||||
| import com.squareup.kotlinpoet.ksp.writeTo | 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.GenerateGenericKoinDefinition | ||||||
| import dev.inmo.micro_utils.koin.annotations.GenerateKoinDefinition | import dev.inmo.micro_utils.koin.annotations.GenerateKoinDefinition | ||||||
| import org.koin.core.Koin | import org.koin.core.Koin | ||||||
| @@ -237,15 +238,7 @@ class Processor( | |||||||
|                     """.trimIndent() |                     """.trimIndent() | ||||||
|                 ) |                 ) | ||||||
|                 ksFile.getAnnotationsByType(GenerateKoinDefinition::class).forEach { |                 ksFile.getAnnotationsByType(GenerateKoinDefinition::class).forEach { | ||||||
|                     val type = runCatching { |                     val type = safeClassName { it.type } | ||||||
|                         it.type.asTypeName() |  | ||||||
|                     }.getOrElse { e -> |  | ||||||
|                         if (e is KSTypeNotPresentException) { |  | ||||||
|                             e.ksType.toClassName() |  | ||||||
|                         } else { |  | ||||||
|                             throw e |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|                     val targetType = runCatching { |                     val targetType = runCatching { | ||||||
|                         type.parameterizedBy(*(it.typeArgs.takeIf { it.isNotEmpty() } ?.map { it.asTypeName() } ?.toTypedArray() ?: return@runCatching type)) |                         type.parameterizedBy(*(it.typeArgs.takeIf { it.isNotEmpty() } ?.map { it.asTypeName() } ?.toTypedArray() ?: return@runCatching type)) | ||||||
|                     }.getOrElse { e -> |                     }.getOrElse { e -> | ||||||
|   | |||||||
| @@ -5,7 +5,6 @@ package dev.inmo.micro_utils.koin.generator.test | |||||||
|  |  | ||||||
| import kotlin.Any | import kotlin.Any | ||||||
| import kotlin.Boolean | import kotlin.Boolean | ||||||
| import kotlin.Deprecated |  | ||||||
| import kotlin.String | import kotlin.String | ||||||
| import org.koin.core.Koin | import org.koin.core.Koin | ||||||
| import org.koin.core.definition.Definition | import org.koin.core.definition.Definition | ||||||
| @@ -30,95 +29,59 @@ public val Koin.sampleInfo: Test<String> | |||||||
| /** | /** | ||||||
|  * @return Definition by key "sampleInfo" with [parameters] |  * @return Definition by key "sampleInfo" with [parameters] | ||||||
|  */ |  */ | ||||||
| public inline fun Scope.sampleInfo(noinline parameters: ParametersDefinition): Test<String> = | public inline fun Scope.sampleInfo(noinline parameters: ParametersDefinition): Test<String> = get(named("sampleInfo"), parameters) | ||||||
|     get(named("sampleInfo"), parameters) |  | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * @return Definition by key "sampleInfo" with [parameters] |  * @return Definition by key "sampleInfo" with [parameters] | ||||||
|  */ |  */ | ||||||
| public inline fun Koin.sampleInfo(noinline parameters: ParametersDefinition): Test<String> = | public inline fun Koin.sampleInfo(noinline parameters: ParametersDefinition): Test<String> = get(named("sampleInfo"), parameters) | ||||||
|     get(named("sampleInfo"), parameters) |  | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Will register [definition] with [org.koin.core.module.Module.single] and key "sampleInfo" |  * Will register [definition] with [org.koin.core.module.Module.single] and key "sampleInfo" | ||||||
|  */ |  */ | ||||||
| @Deprecated( | public fun Module.singleSampleInfo(createdAtStart: Boolean = false, definition: Definition<Test<String>>): KoinDefinition<Test<String>> = single(named("sampleInfo"), createdAtStart = createdAtStart, definition = definition) | ||||||
|   "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" |  * Will register [definition] with [org.koin.core.module.Module.factory] and key "sampleInfo" | ||||||
|  */ |  */ | ||||||
| @Deprecated( | public fun Module.factorySampleInfo(definition: Definition<Test<String>>): KoinDefinition<Test<String>> = factory(named("sampleInfo"), definition = definition) | ||||||
|   "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] |  * @return Definition by key "test" with [parameters] | ||||||
|  */ |  */ | ||||||
| public inline fun <reified T : Any> Scope.test(noinline parameters: ParametersDefinition? = null): T | public inline fun <reified T : Any> Scope.test(noinline parameters: ParametersDefinition? = null): T = get(named("test"), parameters) | ||||||
|     = get(named("test"), parameters) |  | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * @return Definition by key "test" with [parameters] |  * @return Definition by key "test" with [parameters] | ||||||
|  */ |  */ | ||||||
| public inline fun <reified T : Any> Koin.test(noinline parameters: ParametersDefinition? = null): T | public inline fun <reified T : Any> Koin.test(noinline parameters: ParametersDefinition? = null): T = get(named("test"), parameters) | ||||||
|     = get(named("test"), parameters) |  | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Will register [definition] with [org.koin.core.module.Module.single] and key "test" |  * 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 | public inline fun <reified T : Any> Module.singleTest(createdAtStart: Boolean = false, noinline definition: Definition<T>): KoinDefinition<T> = single(named("test"), createdAtStart = createdAtStart, definition = definition) | ||||||
|     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" |  * 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>): | public inline fun <reified T : Any> Module.factoryTest(noinline definition: Definition<T>): KoinDefinition<T> = factory(named("test"), definition = definition) | ||||||
|     KoinDefinition<T> = factory(named("test"), definition = definition) |  | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * @return Definition by key "testNullable" with [parameters] |  * @return Definition by key "testNullable" with [parameters] | ||||||
|  */ |  */ | ||||||
| public inline fun <reified T : Any> Scope.testNullable(noinline parameters: ParametersDefinition? = | public inline fun <reified T : Any> Scope.testNullable(noinline parameters: ParametersDefinition? = null): T? = getOrNull(named("testNullable"), parameters) | ||||||
|     null): T? = getOrNull(named("testNullable"), parameters) |  | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * @return Definition by key "testNullable" with [parameters] |  * @return Definition by key "testNullable" with [parameters] | ||||||
|  */ |  */ | ||||||
| public inline fun <reified T : Any> Koin.testNullable(noinline parameters: ParametersDefinition? = | public inline fun <reified T : Any> Koin.testNullable(noinline parameters: ParametersDefinition? = null): T? = getOrNull(named("testNullable"), parameters) | ||||||
|     null): T? = getOrNull(named("testNullable"), parameters) |  | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Will register [definition] with [org.koin.core.module.Module.single] and key "testNullable" |  * Will register [definition] with [org.koin.core.module.Module.single] and key "testNullable" | ||||||
|  */ |  */ | ||||||
| public inline fun <reified T : Any> Module.singleTestNullable(createdAtStart: Boolean = false, | public inline fun <reified T : Any> Module.singleTestNullable(createdAtStart: Boolean = false, noinline definition: Definition<T>): KoinDefinition<T> = single(named("testNullable"), createdAtStart = createdAtStart, definition = definition) | ||||||
|     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" |  * 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>): | public inline fun <reified T : Any> Module.factoryTestNullable(noinline definition: Definition<T>): KoinDefinition<T> = factory(named("testNullable"), definition = definition) | ||||||
|     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 | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -1,9 +1,8 @@ | |||||||
| package dev.inmo.micro_utils.pagination.compose | package dev.inmo.micro_utils.pagination.compose | ||||||
|  |  | ||||||
| import androidx.compose.runtime.* | import androidx.compose.runtime.* | ||||||
| import dev.inmo.micro_utils.coroutines.SpecialMutableStateFlow | import dev.inmo.micro_utils.coroutines.MutableRedeliverStateFlow | ||||||
| import dev.inmo.micro_utils.coroutines.launchLoggingDropExceptions | import dev.inmo.micro_utils.coroutines.launchLoggingDropExceptions | ||||||
| import dev.inmo.micro_utils.coroutines.runCatchingLogging |  | ||||||
| import dev.inmo.micro_utils.pagination.* | import dev.inmo.micro_utils.pagination.* | ||||||
| import kotlinx.coroutines.CoroutineScope | import kotlinx.coroutines.CoroutineScope | ||||||
| import kotlinx.coroutines.Job | import kotlinx.coroutines.Job | ||||||
| @@ -27,8 +26,8 @@ class InfinityPagedComponentContext<T> internal constructor( | |||||||
|     private val loader: suspend InfinityPagedComponentContext<T>.(Pagination) -> PaginationResult<T> |     private val loader: suspend InfinityPagedComponentContext<T>.(Pagination) -> PaginationResult<T> | ||||||
| ) { | ) { | ||||||
|     internal val startPage = SimplePagination(page, size) |     internal val startPage = SimplePagination(page, size) | ||||||
|     internal val latestLoadedPage = SpecialMutableStateFlow<PaginationResult<T>?>(null) |     internal val latestLoadedPage = MutableRedeliverStateFlow<PaginationResult<T>?>(null) | ||||||
|     internal val dataState = SpecialMutableStateFlow<List<T>?>(null) |     internal val dataState = MutableRedeliverStateFlow<List<T>?>(null) | ||||||
|     internal var loadingJob: Job? = null |     internal var loadingJob: Job? = null | ||||||
|     internal val loadingMutex = Mutex() |     internal val loadingMutex = Mutex() | ||||||
|  |  | ||||||
| @@ -65,6 +64,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. |  * Composable function for managing an infinitely paged component. | ||||||
|  * |  * | ||||||
| @@ -76,7 +129,7 @@ class InfinityPagedComponentContext<T> internal constructor( | |||||||
|  * receive null as `it` parameter |  * receive null as `it` parameter | ||||||
|  */ |  */ | ||||||
| @Composable | @Composable | ||||||
| internal fun <T> InfinityPagedComponent( | fun <T> InfinityPagedComponent( | ||||||
|     page: Int, |     page: Int, | ||||||
|     size: Int, |     size: Int, | ||||||
|     loader: suspend InfinityPagedComponentContext<T>.(Pagination) -> PaginationResult<T>, |     loader: suspend InfinityPagedComponentContext<T>.(Pagination) -> PaginationResult<T>, | ||||||
| @@ -84,13 +137,8 @@ internal fun <T> InfinityPagedComponent( | |||||||
|     block: @Composable InfinityPagedComponentContext<T>.(List<T>?) -> Unit |     block: @Composable InfinityPagedComponentContext<T>.(List<T>?) -> Unit | ||||||
| ) { | ) { | ||||||
|     val scope = predefinedScope ?: rememberCoroutineScope() |     val scope = predefinedScope ?: rememberCoroutineScope() | ||||||
|     val context = remember { InfinityPagedComponentContext<T>(page, size, scope, loader) } |     val context = rememberInfinityPagedComponentContext(page = page, size = size, scope = scope, loader = loader) | ||||||
|     remember { |     InfinityPagedComponent(context, block) | ||||||
|         context.reload() |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     val dataState = context.dataState.collectAsState() |  | ||||||
|     context.block(dataState.value) |  | ||||||
| } | } | ||||||
|  |  | ||||||
| /** | /** | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| package dev.inmo.micro_utils.pagination.compose | package dev.inmo.micro_utils.pagination.compose | ||||||
|  |  | ||||||
| import androidx.compose.runtime.* | import androidx.compose.runtime.* | ||||||
| import dev.inmo.micro_utils.coroutines.SpecialMutableStateFlow | import dev.inmo.micro_utils.coroutines.MutableRedeliverStateFlow | ||||||
| import dev.inmo.micro_utils.coroutines.launchLoggingDropExceptions | import dev.inmo.micro_utils.coroutines.launchLoggingDropExceptions | ||||||
| import dev.inmo.micro_utils.pagination.* | import dev.inmo.micro_utils.pagination.* | ||||||
| import kotlinx.coroutines.CoroutineScope | import kotlinx.coroutines.CoroutineScope | ||||||
| @@ -28,8 +28,8 @@ class PagedComponentContext<T> internal constructor( | |||||||
|     private val loader: suspend PagedComponentContext<T>.(Pagination) -> PaginationResult<T> |     private val loader: suspend PagedComponentContext<T>.(Pagination) -> PaginationResult<T> | ||||||
| ) { | ) { | ||||||
|     internal val startPage = SimplePagination(initialPage, size) |     internal val startPage = SimplePagination(initialPage, size) | ||||||
|     internal val latestLoadedPage = SpecialMutableStateFlow<PaginationResult<T>?>(null) |     internal val latestLoadedPage = MutableRedeliverStateFlow<PaginationResult<T>?>(null) | ||||||
|     internal val dataState = SpecialMutableStateFlow<PaginationResult<T>?>(null) |     internal val dataState = MutableRedeliverStateFlow<PaginationResult<T>?>(null) | ||||||
|     internal var loadingJob: Job? = null |     internal var loadingJob: Job? = null | ||||||
|     internal val loadingMutex = Mutex() |     internal val loadingMutex = Mutex() | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,10 +1,81 @@ | |||||||
| package dev.inmo.micro_utils.repos.exposed | package dev.inmo.micro_utils.repos.exposed | ||||||
|  |  | ||||||
| import org.jetbrains.exposed.sql.* | import org.jetbrains.exposed.sql.* | ||||||
|  | import org.jetbrains.exposed.sql.SchemaUtils.addMissingColumnsStatements | ||||||
|  | import org.jetbrains.exposed.sql.SchemaUtils.checkMappingConsistence | ||||||
|  | import org.jetbrains.exposed.sql.SchemaUtils.createStatements | ||||||
|  | import org.jetbrains.exposed.sql.transactions.TransactionManager | ||||||
| import org.jetbrains.exposed.sql.transactions.transaction | import org.jetbrains.exposed.sql.transactions.transaction | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Code in this function mostly duplicates Exposed [SchemaUtils.createMissingTablesAndColumns]. It made due to deprecation | ||||||
|  |  * status of the last one and potential lost of it in future updates. | ||||||
|  |  * | ||||||
|  |  * Code doing: | ||||||
|  |  * | ||||||
|  |  * * Creating missed tables | ||||||
|  |  * * Altering missed tables (where possible) | ||||||
|  |  * * Calculate problems with [checkMappingConsistence] and add them in execution too | ||||||
|  |  * | ||||||
|  |  * All changes made in [transaction] with [database] as its `db` argument | ||||||
|  |  */ | ||||||
|  | fun initTablesInTransaction(vararg tables: Table, database: Database, inBatch: Boolean = false, withLogs: Boolean = true) { | ||||||
|  |     transaction(database) { | ||||||
|  |         fun <R> logTimeSpent(message: String, withLogs: Boolean, block: () -> R): R { | ||||||
|  |             return if (withLogs) { | ||||||
|  |                 val start = System.currentTimeMillis() | ||||||
|  |                 val answer = block() | ||||||
|  |                 exposedLogger.info(message + " took " + (System.currentTimeMillis() - start) + "ms") | ||||||
|  |                 answer | ||||||
|  |             } else { | ||||||
|  |                 block() | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         fun Transaction.execStatements(inBatch: Boolean, statements: List<String>) { | ||||||
|  |             if (inBatch) { | ||||||
|  |                 execInBatch(statements) | ||||||
|  |             } else { | ||||||
|  |                 for (statement in statements) { | ||||||
|  |                     exec(statement) | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         with(TransactionManager.current()) { | ||||||
|  |             db.dialect.resetCaches() | ||||||
|  |             val createStatements = logTimeSpent("Preparing create tables statements", withLogs) { | ||||||
|  |                 createStatements(*tables) | ||||||
|  |             } | ||||||
|  |             logTimeSpent("Executing create tables statements", withLogs) { | ||||||
|  |                 execStatements(inBatch, createStatements) | ||||||
|  |                 commit() | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             val alterStatements = logTimeSpent("Preparing alter table statements", withLogs) { | ||||||
|  |                 addMissingColumnsStatements(tables = tables, withLogs) | ||||||
|  |             } | ||||||
|  |             logTimeSpent("Executing alter table statements", withLogs) { | ||||||
|  |                 execStatements(inBatch, alterStatements) | ||||||
|  |                 commit() | ||||||
|  |             } | ||||||
|  |             val executedStatements = createStatements + alterStatements | ||||||
|  |             logTimeSpent("Checking mapping consistence", withLogs) { | ||||||
|  |                 val modifyTablesStatements = checkMappingConsistence( | ||||||
|  |                     tables = tables, | ||||||
|  |                     withLogs | ||||||
|  |                 ).filter { it !in executedStatements } | ||||||
|  |                 execStatements(inBatch, modifyTablesStatements) | ||||||
|  |                 commit() | ||||||
|  |             } | ||||||
|  |             db.dialect.resetCaches() | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fun Table.initTable(database: Database, inBatch: Boolean, withLogs: Boolean) { | ||||||
|  |     initTablesInTransaction(this, database = database, inBatch = inBatch, withLogs = withLogs) | ||||||
|  | } | ||||||
| fun Table.initTable(database: Database) { | fun Table.initTable(database: Database) { | ||||||
|     transaction(database) { SchemaUtils.createMissingTablesAndColumns(this@initTable) } |     initTable(database = database, inBatch = false, withLogs = true) | ||||||
| } | } | ||||||
|  |  | ||||||
| fun <T> T.initTable() where T: ExposedRepo, T: Table = initTable(database) | fun <T> T.initTable() where T: ExposedRepo, T: Table = initTable(database) | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user