mirror of
				https://github.com/InsanusMokrassar/MicroUtils.git
				synced 2025-10-26 09:40:26 +00:00 
			
		
		
		
	Compare commits
	
		
			66 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| f4af28059b | |||
| c1476bd075 | |||
| 16c720fddd | |||
| 8b4b4a5eca | |||
| 32e6e5b7e2 | |||
| a9f7fd8e32 | |||
| 95be1a26f2 | |||
| ef9b31aee0 | |||
| df3c01ff0a | |||
| 4704c5a33d | |||
| 225c06550a | |||
| f0987614c6 | |||
| 269c2876f3 | |||
| 168d6acf7c | |||
| a5f718e257 | |||
| 4f68459582 | |||
| 442db122cf | |||
| 580d757be2 | |||
| 47b0f6d2d8 | |||
| 3f6f6ebc2b | |||
| 2645ea29d6 | |||
| 79f2041565 | |||
| 4a7567f288 | |||
| 8a890ed6ed | |||
| 3d90df6897 | |||
| 681c13144a | |||
| b64f2e6d32 | |||
| 428eabb1bd | |||
| 2162e83bce | |||
| 6142022283 | |||
| e6d9c8250f | |||
| 46178e723b | |||
| 605f55acd2 | |||
| 0f8b69aa60 | |||
| 551d8ec480 | |||
| fc48446ec4 | |||
| 3644b83ac6 | |||
| cd73791b6f | |||
| 03de71df2e | |||
| 83d5d3faf4 | |||
| 0c8bec4c89 | |||
| 7fc93817c1 | |||
| d0a00031a1 | |||
| 6ebc5aa0c2 | |||
| 8a6b4bb49e | |||
| 20799b9a3e | |||
| ec3afc615c | |||
| da692ccfc3 | |||
| 53b89f3a18 | |||
| 58cded28d3 | |||
| 592c5f3732 | |||
| f44a78a5f5 | |||
| e0bdd5dfdc | |||
| 99c0f06b72 | |||
| 66fc6df3d7 | |||
| a36425a905 | |||
| d920fee6d4 | |||
| 23590be5de | |||
| 94acc3c93b | |||
| 5616326a3b | |||
| 7601860c5c | |||
| 8b43d785cc | |||
| b62d3a0b7d | |||
| fad73c7213 | |||
| 2403c7c2b0 | |||
| 204955bcce | 
							
								
								
									
										82
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										82
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -1,9 +1,89 @@ | ||||
| # Changelog | ||||
|  | ||||
| ## 0.16.3 | ||||
| ## 0.16.10 | ||||
|  | ||||
| * `Repos`: | ||||
|     * `Cache`: | ||||
|         * New transformer type: `ReadCRUDFromKeyValueRepo` | ||||
|         * New transformer type: `ReadKeyValueFromCRUDRepo` | ||||
| * `Pagination`: | ||||
|     * New `paginate` extensions with `reversed` support for `List`/`Set` | ||||
|  | ||||
| ## 0.16.9 | ||||
|  | ||||
| * `Versions`: | ||||
|     * `Koin`: `3.2.2` -> `3.3.2` | ||||
|     * `AppCompat`: `1.5.1` -> `1.6.0` | ||||
| * `Ktor`: | ||||
|     * `Client` | ||||
|         * `HttpResponse.bodyOrNull` now retrieve callback to check if body should be received or null | ||||
|         * New extension `HttpResponse.bodyOrNullOnNoContent` | ||||
|  | ||||
| ## 0.16.8 | ||||
|  | ||||
| * `Versions`: | ||||
|     * `Ktor`: `2.2.2` -> `2.2.3` | ||||
| * `Ktor`: | ||||
|     * `Client` | ||||
|         * Fixes in `HttpClient.uniUpload` | ||||
|     * `Server` | ||||
|         * Fixes in `PartData.FileItem.download` | ||||
| * `Repos`: | ||||
|     * `Cache`: | ||||
|         * New type of caches: `FallbackCacheRepo` | ||||
|         * Fixes in `Write*` variants of cached repos | ||||
|         * New type `ActionWrapper` | ||||
|         * New `AutoRecache*` classes for all types of repos as `FallbackCacheRepo`s | ||||
|     * `Common`: | ||||
|         * New transformations for key-value and key-values vice-verse | ||||
|         * Fixes in `FileReadKeyValueRepo` | ||||
|  | ||||
| ## 0.16.7 | ||||
|  | ||||
| * `Common`: | ||||
|     * New extensions `ifTrue`/`ifFalse`/`alsoIfTrue`/`alsoIfFalse`/`letIfTrue`/`letIfFalse` | ||||
|     * `Diff` now is serializable | ||||
|     * Add `IndexedValue` serializer | ||||
|     * `repeatOnFailure` extending: now you may pass any lambda to check if continue to try/do something | ||||
|     * `Compose`: | ||||
|         * New extension `MutableState.asState` | ||||
| * `Coroutines`: | ||||
|     * `Compose`: | ||||
|         * All the `Flow` conversations to compose `State`/`MutableState`/`SnapshotStateList`/`List` got several new | ||||
|         parameters | ||||
|         * `Flow.toMutableState` now is deprecated in favor to `asMutableComposeState` | ||||
| * `Repos`: | ||||
|     * `Cache`: | ||||
|         * New type `FullCacheRepo` | ||||
|         * New type `CommonCacheRepo` | ||||
|         * `CacheRepo` got `invalidate` method. It will fully reload `FullCacheRepo` and just clear `CommonCacheRepo` | ||||
|         * New extensions `KVCache.actualizeAll` | ||||
|  | ||||
| ## 0.16.6 | ||||
|  | ||||
| * `Startup`: | ||||
|     * `Launcher`: | ||||
|         * Improvements in `StartLauncherPlugin#start` methods | ||||
|         * Add opportunity to pass second argument on `JVM` platform as log level | ||||
| * `Repos`: | ||||
|     * `Ktor`: | ||||
|         * `Client`: | ||||
|             * All clients repos got opportunity to customize their flows | ||||
|     * `Exposed`: | ||||
|         * Extensions `eqOrIsNull` and `neqOrIsNotNull` for `Column` | ||||
|  | ||||
| ## 0.16.5 | ||||
|  | ||||
| * `Versions`: | ||||
|     * `Ktor`: `2.2.1` -> `2.2.2` | ||||
|  | ||||
| ## 0.16.4 | ||||
|  | ||||
| * `Coroutines`: | ||||
|     * Create `launchInCurrentThread` | ||||
|  | ||||
| ## 0.16.3 | ||||
|  | ||||
| * `Startup`: | ||||
|     * `Launcher`: | ||||
|         * All starting API have been moved into `StartLauncherPlugin` and do not require serialize/deserialize cycle for now | ||||
|   | ||||
| @@ -0,0 +1,10 @@ | ||||
| package dev.inmo.micro_utils.common.compose | ||||
|  | ||||
| import androidx.compose.runtime.MutableState | ||||
| import androidx.compose.runtime.State | ||||
| import androidx.compose.runtime.derivedStateOf | ||||
|  | ||||
| /** | ||||
|  * Converts current [MutableState] to immutable [State] using [derivedStateOf] | ||||
|  */ | ||||
| fun <T> MutableState<T>.asState(): State<T> = derivedStateOf { this.value } | ||||
| @@ -2,6 +2,8 @@ | ||||
|  | ||||
| package dev.inmo.micro_utils.common | ||||
|  | ||||
| import kotlinx.serialization.Serializable | ||||
|  | ||||
| private inline fun <T> getObject( | ||||
|     additional: MutableList<T>, | ||||
|     iterator: Iterator<T> | ||||
| @@ -24,13 +26,14 @@ private inline fun <T> getObject( | ||||
|  * | ||||
|  * @see calculateDiff | ||||
|  */ | ||||
| @Serializable | ||||
| data class Diff<T> internal constructor( | ||||
|     val removed: List<IndexedValue<T>>, | ||||
|     val removed: List<@Serializable(IndexedValueSerializer::class) IndexedValue<T>>, | ||||
|     /** | ||||
|      * Old-New values pairs | ||||
|      */ | ||||
|     val replaced: List<Pair<IndexedValue<T>, IndexedValue<T>>>, | ||||
|     val added: List<IndexedValue<T>> | ||||
|     val replaced: List<Pair<@Serializable(IndexedValueSerializer::class) IndexedValue<T>, @Serializable(IndexedValueSerializer::class) IndexedValue<T>>>, | ||||
|     val added: List<@Serializable(IndexedValueSerializer::class) IndexedValue<T>> | ||||
| ) | ||||
|  | ||||
| private inline fun <T> performChanges( | ||||
|   | ||||
| @@ -0,0 +1,43 @@ | ||||
| package dev.inmo.micro_utils.common | ||||
|  | ||||
| inline fun <T> Boolean.letIfTrue(block: () -> T): T? { | ||||
|     return if (this) { | ||||
|         block() | ||||
|     } else { | ||||
|         null | ||||
|     } | ||||
| } | ||||
|  | ||||
| inline fun <T> Boolean.letIfFalse(block: () -> T): T? { | ||||
|     return if (this) { | ||||
|         null | ||||
|     } else { | ||||
|         block() | ||||
|     } | ||||
| } | ||||
|  | ||||
| inline fun Boolean.alsoIfTrue(block: () -> Unit): Boolean { | ||||
|     letIfTrue(block) | ||||
|     return this | ||||
| } | ||||
|  | ||||
| inline fun Boolean.alsoIfFalse(block: () -> Unit): Boolean { | ||||
|     letIfFalse(block) | ||||
|     return this | ||||
| } | ||||
|  | ||||
| inline fun <T> Boolean.ifTrue(block: () -> T): T? { | ||||
|     return if (this) { | ||||
|         block() | ||||
|     } else { | ||||
|         null | ||||
|     } | ||||
| } | ||||
|  | ||||
| inline fun <T> Boolean.ifFalse(block: () -> T): T? { | ||||
|     return if (this) { | ||||
|         null | ||||
|     } else { | ||||
|         block() | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,30 @@ | ||||
| package dev.inmo.micro_utils.common | ||||
|  | ||||
| import kotlinx.serialization.KSerializer | ||||
| import kotlinx.serialization.Serializer | ||||
| import kotlinx.serialization.builtins.PairSerializer | ||||
| import kotlinx.serialization.builtins.serializer | ||||
| import kotlinx.serialization.descriptors.SerialDescriptor | ||||
| import kotlinx.serialization.encoding.Decoder | ||||
| import kotlinx.serialization.encoding.Encoder | ||||
|  | ||||
| class IndexedValueSerializer<T>(private val subSerializer: KSerializer<T>) : KSerializer<IndexedValue<T>> { | ||||
|     private val originalSerializer = PairSerializer(Int.serializer(), subSerializer) | ||||
|     override val descriptor: SerialDescriptor | ||||
|         get() = originalSerializer.descriptor | ||||
|  | ||||
|     override fun deserialize(decoder: Decoder): IndexedValue<T> { | ||||
|         val pair = originalSerializer.deserialize(decoder) | ||||
|         return IndexedValue( | ||||
|             pair.first, | ||||
|             pair.second | ||||
|         ) | ||||
|     } | ||||
|  | ||||
|     override fun serialize(encoder: Encoder, value: IndexedValue<T>) { | ||||
|         originalSerializer.serialize( | ||||
|             encoder, | ||||
|             Pair(value.index, value.value) | ||||
|         ) | ||||
|     } | ||||
| } | ||||
| @@ -1,5 +1,27 @@ | ||||
| package dev.inmo.micro_utils.common | ||||
|  | ||||
| /** | ||||
|  * Executes the given [action] until getting of successful result specified number of [times]. | ||||
|  * | ||||
|  * A zero-based index of current iteration is passed as a parameter to [action]. | ||||
|  */ | ||||
| inline fun <R> repeatOnFailure( | ||||
|     onFailure: (Throwable) -> Boolean, | ||||
|     action: () -> R | ||||
| ): Result<R> { | ||||
|     do { | ||||
|         runCatching { | ||||
|             action() | ||||
|         }.onFailure { | ||||
|             if (!onFailure(it)) { | ||||
|                 return Result.failure(it) | ||||
|             } | ||||
|         }.onSuccess { | ||||
|             return Result.success(it) | ||||
|         } | ||||
|     } while (true) | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Executes the given [action] until getting of successful result specified number of [times]. | ||||
|  * | ||||
| @@ -10,12 +32,23 @@ inline fun <R> repeatOnFailure( | ||||
|     onEachFailure: (Throwable) -> Unit = {}, | ||||
|     action: (Int) -> R | ||||
| ): Optional<R> { | ||||
|     repeat(times) { | ||||
|         runCatching { | ||||
|             action(it) | ||||
|         }.onFailure(onEachFailure).onSuccess { | ||||
|             return Optional.presented(it) | ||||
|     var i = 0 | ||||
|     val result = repeatOnFailure( | ||||
|         { | ||||
|             onEachFailure(it) | ||||
|             if (i < times) { | ||||
|                 i++ | ||||
|                 true | ||||
|             } else { | ||||
|                 false | ||||
|             } | ||||
|         } | ||||
|     return Optional.absent() | ||||
|     ) { | ||||
|         action(i) | ||||
|     } | ||||
|     return if (result.isSuccess) { | ||||
|         Optional.presented(result.getOrThrow()) | ||||
|     } else { | ||||
|         Optional.absent() | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -3,24 +3,58 @@ package dev.inmo.micro_utils.coroutines.compose | ||||
| import androidx.compose.runtime.* | ||||
| import androidx.compose.runtime.snapshots.SnapshotStateList | ||||
| import dev.inmo.micro_utils.common.applyDiff | ||||
| import dev.inmo.micro_utils.coroutines.ExceptionHandler | ||||
| import dev.inmo.micro_utils.coroutines.defaultSafelyWithoutExceptionHandlerWithNull | ||||
| import dev.inmo.micro_utils.coroutines.subscribeSafelyWithoutExceptions | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlinx.coroutines.Dispatchers | ||||
| import kotlinx.coroutines.flow.Flow | ||||
| import kotlinx.coroutines.flow.StateFlow | ||||
| import kotlinx.coroutines.withContext | ||||
| import kotlin.coroutines.CoroutineContext | ||||
|  | ||||
| /** | ||||
|  * Each value of [this] [Flow] will trigger [applyDiff] to the result [SnapshotStateList] | ||||
|  * | ||||
|  * @param scope Will be used to [subscribeSafelyWithoutExceptions] on [this] to update returned [SnapshotStateList] | ||||
|  * @param useContextOnChange Will be used to change context inside of [subscribeSafelyWithoutExceptions] to ensure that | ||||
|  * change will happen in the required [CoroutineContext]. [Dispatchers.Main] by default | ||||
|  * @param onException Will be passed to the [subscribeSafelyWithoutExceptions] as uncaught exceptions handler | ||||
|  */ | ||||
| @Suppress("NOTHING_TO_INLINE") | ||||
| inline fun <reified T> Flow<List<T>>.asMutableComposeListState( | ||||
|     scope: CoroutineScope | ||||
|     scope: CoroutineScope, | ||||
|     useContextOnChange: CoroutineContext? = Dispatchers.Main, | ||||
|     noinline onException: ExceptionHandler<List<T>?> = defaultSafelyWithoutExceptionHandlerWithNull, | ||||
| ): SnapshotStateList<T> { | ||||
|     val state = mutableStateListOf<T>() | ||||
|     subscribeSafelyWithoutExceptions(scope) { | ||||
|     val changeBlock: suspend (List<T>) -> Unit = useContextOnChange ?.let { | ||||
|         { | ||||
|             withContext(useContextOnChange) { | ||||
|                 state.applyDiff(it) | ||||
|             } | ||||
|         } | ||||
|     } ?: { | ||||
|         state.applyDiff(it) | ||||
|     } | ||||
|     subscribeSafelyWithoutExceptions(scope, onException, changeBlock) | ||||
|     return state | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * In fact, it is just classcast of [asMutableComposeListState] to [List] | ||||
|  * | ||||
|  * @param scope Will be used to [subscribeSafelyWithoutExceptions] on [this] to update returned [List] | ||||
|  * @param useContextOnChange Will be used to change context inside of [subscribeSafelyWithoutExceptions] to ensure that | ||||
|  * change will happen in the required [CoroutineContext]. [Dispatchers.Main] by default | ||||
|  * @param onException Will be passed to the [subscribeSafelyWithoutExceptions] as uncaught exceptions handler | ||||
|  * | ||||
|  * @return Changing in time [List] which follow [Flow] values | ||||
|  */ | ||||
| @Suppress("NOTHING_TO_INLINE") | ||||
| inline fun <reified T> Flow<List<T>>.asComposeList( | ||||
|     scope: CoroutineScope | ||||
| ): List<T> = asMutableComposeListState(scope) | ||||
|     scope: CoroutineScope, | ||||
|     useContextOnChange: CoroutineContext? = Dispatchers.Main, | ||||
|     noinline onException: ExceptionHandler<List<T>?> = defaultSafelyWithoutExceptionHandlerWithNull, | ||||
| ): List<T> = asMutableComposeListState(scope, useContextOnChange, onException) | ||||
|  | ||||
|   | ||||
| @@ -1,35 +1,94 @@ | ||||
| package dev.inmo.micro_utils.coroutines.compose | ||||
|  | ||||
| import androidx.compose.runtime.* | ||||
| import dev.inmo.micro_utils.common.compose.asState | ||||
| import dev.inmo.micro_utils.coroutines.ExceptionHandler | ||||
| import dev.inmo.micro_utils.coroutines.defaultSafelyWithoutExceptionHandlerWithNull | ||||
| import dev.inmo.micro_utils.coroutines.doInUI | ||||
| import dev.inmo.micro_utils.coroutines.subscribeSafelyWithoutExceptions | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlinx.coroutines.Dispatchers | ||||
| import kotlinx.coroutines.flow.Flow | ||||
| import kotlinx.coroutines.flow.StateFlow | ||||
| import kotlinx.coroutines.withContext | ||||
| import kotlin.coroutines.CoroutineContext | ||||
|  | ||||
| /** | ||||
|  * Will map [this] [Flow] as [MutableState]. Returned [MutableState] WILL NOT change source [Flow] | ||||
|  * | ||||
|  * @param initial First value which will be passed to the result [MutableState] | ||||
|  * @param scope Will be used to [subscribeSafelyWithoutExceptions] on [this] to update returned [MutableState] | ||||
|  * @param useContextOnChange Will be used to change context inside of [subscribeSafelyWithoutExceptions] to ensure that | ||||
|  * change will happen in the required [CoroutineContext]. [Dispatchers.Main] by default | ||||
|  * @param onException Will be passed to the [subscribeSafelyWithoutExceptions] as uncaught exceptions handler | ||||
|  */ | ||||
| fun <T> Flow<T>.asMutableComposeState( | ||||
|     initial: T, | ||||
|     scope: CoroutineScope | ||||
|     scope: CoroutineScope, | ||||
|     useContextOnChange: CoroutineContext? = Dispatchers.Main, | ||||
|     onException: ExceptionHandler<T?> = defaultSafelyWithoutExceptionHandlerWithNull, | ||||
| ): MutableState<T> { | ||||
|     val state = mutableStateOf(initial) | ||||
|     subscribeSafelyWithoutExceptions(scope) { state.value = it } | ||||
|     val changeBlock: suspend (T) -> Unit = useContextOnChange ?.let { | ||||
|         { | ||||
|             withContext(useContextOnChange) { | ||||
|                 state.value = it | ||||
|             } | ||||
|         } | ||||
|     } ?: { | ||||
|         state.value = it | ||||
|     } | ||||
|     subscribeSafelyWithoutExceptions(scope, onException, block = changeBlock) | ||||
|     return state | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Will map [this] [StateFlow] as [MutableState]. Returned [MutableState] WILL NOT change source [StateFlow]. | ||||
|  * This conversation will pass its [StateFlow.value] as the first value | ||||
|  * | ||||
|  * @param scope Will be used to [subscribeSafelyWithoutExceptions] on [this] to update returned [MutableState] | ||||
|  * @param useContextOnChange Will be used to change context inside of [subscribeSafelyWithoutExceptions] to ensure that | ||||
|  * change will happen in the required [CoroutineContext]. [Dispatchers.Main] by default | ||||
|  * @param onException Will be passed to the [subscribeSafelyWithoutExceptions] as uncaught exceptions handler | ||||
|  */ | ||||
| @Suppress("NOTHING_TO_INLINE") | ||||
| inline fun <T> StateFlow<T>.asMutableComposeState( | ||||
|     scope: CoroutineScope | ||||
| ): MutableState<T> = asMutableComposeState(value, scope) | ||||
|     scope: CoroutineScope, | ||||
|     useContextOnChange: CoroutineContext? = Dispatchers.Main, | ||||
|     noinline onException: ExceptionHandler<T?> = defaultSafelyWithoutExceptionHandlerWithNull, | ||||
| ): MutableState<T> = asMutableComposeState(value, scope, useContextOnChange, onException) | ||||
|  | ||||
| /** | ||||
|  * Will create [MutableState] using [asMutableComposeState] and use [asState] to convert it as immutable state | ||||
|  * | ||||
|  * @param initial First value which will be passed to the result [State] | ||||
|  * @param scope Will be used to [subscribeSafelyWithoutExceptions] on [this] to update returned [State] | ||||
|  * @param useContextOnChange Will be used to change context inside of [subscribeSafelyWithoutExceptions] to ensure that | ||||
|  * change will happen in the required [CoroutineContext]. [Dispatchers.Main] by default | ||||
|  * @param onException Will be passed to the [subscribeSafelyWithoutExceptions] as uncaught exceptions handler | ||||
|  */ | ||||
| fun <T> Flow<T>.asComposeState( | ||||
|     initial: T, | ||||
|     scope: CoroutineScope | ||||
|     scope: CoroutineScope, | ||||
|     useContextOnChange: CoroutineContext? = Dispatchers.Main, | ||||
|     onException: ExceptionHandler<T?> = defaultSafelyWithoutExceptionHandlerWithNull, | ||||
| ): State<T> { | ||||
|     val state = asMutableComposeState(initial, scope) | ||||
|     return derivedStateOf { state.value } | ||||
|     val state = asMutableComposeState(initial, scope, useContextOnChange, onException) | ||||
|     return state.asState() | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Will map [this] [StateFlow] as [State]. This conversation will pass its [StateFlow.value] as the first value | ||||
|  * | ||||
|  * @param scope Will be used to [subscribeSafelyWithoutExceptions] on [this] to update returned [State] | ||||
|  * @param useContextOnChange Will be used to change context inside of [subscribeSafelyWithoutExceptions] to ensure that | ||||
|  * change will happen in the required [CoroutineContext]. [Dispatchers.Main] by default | ||||
|  * @param onException Will be passed to the [subscribeSafelyWithoutExceptions] as uncaught exceptions handler | ||||
|  */ | ||||
| @Suppress("NOTHING_TO_INLINE") | ||||
| inline fun <T> StateFlow<T>.asComposeState( | ||||
|     scope: CoroutineScope | ||||
| ): State<T> = asComposeState(value, scope) | ||||
|     scope: CoroutineScope, | ||||
|     useContextOnChange: CoroutineContext? = Dispatchers.Main, | ||||
|     noinline onException: ExceptionHandler<T?> = defaultSafelyWithoutExceptionHandlerWithNull, | ||||
| ): State<T> = asComposeState(value, scope, useContextOnChange, onException) | ||||
|  | ||||
|   | ||||
| @@ -7,17 +7,15 @@ import kotlinx.coroutines.CoroutineScope | ||||
| import kotlinx.coroutines.flow.Flow | ||||
| import kotlinx.coroutines.flow.StateFlow | ||||
|  | ||||
| @Deprecated("Duplicated functionality", ReplaceWith("asMutableComposeState(initial, scope)", "dev.inmo.micro_utils.coroutines.compose.asMutableComposeState")) | ||||
| fun <T> Flow<T>.toMutableState( | ||||
|     initial: T, | ||||
|     scope: CoroutineScope | ||||
| ): MutableState<T> { | ||||
|     val state = mutableStateOf(initial) | ||||
|     subscribeSafelyWithoutExceptions(scope) { state.value = it } | ||||
|     return state | ||||
| } | ||||
| ): MutableState<T> = asMutableComposeState(initial, scope) | ||||
|  | ||||
| @Deprecated("Duplicated functionality", ReplaceWith("asMutableComposeState(scope)", "dev.inmo.micro_utils.coroutines.compose.asMutableComposeState")) | ||||
| @Suppress("NOTHING_TO_INLINE") | ||||
| inline fun <T> StateFlow<T>.toMutableState( | ||||
|     scope: CoroutineScope | ||||
| ): MutableState<T> = toMutableState(value, scope) | ||||
| ): MutableState<T> = asMutableComposeState(scope) | ||||
|  | ||||
|   | ||||
| @@ -14,5 +14,5 @@ crypto_js_version=4.1.1 | ||||
| # Project data | ||||
|  | ||||
| group=dev.inmo | ||||
| version=0.16.3 | ||||
| android_code_version=171 | ||||
| version=0.16.10 | ||||
| android_code_version=178 | ||||
|   | ||||
| @@ -13,25 +13,25 @@ jb-dokka = "1.7.20" | ||||
| klock = "3.4.0" | ||||
| uuid = "0.6.0" | ||||
|  | ||||
| ktor = "2.2.1" | ||||
| ktor = "2.2.3" | ||||
|  | ||||
| gh-release = "2.4.1" | ||||
|  | ||||
| koin = "3.2.2" | ||||
| koin = "3.3.2" | ||||
|  | ||||
| android-gradle = "7.3.0" | ||||
| dexcount = "3.1.0" | ||||
|  | ||||
| android-coreKtx = "1.9.0" | ||||
| android-recyclerView = "1.2.1" | ||||
| android-appCompat = "1.5.1" | ||||
| android-appCompat = "1.6.0" | ||||
| android-fragment = "1.5.5" | ||||
| android-espresso = "3.4.0" | ||||
| android-test = "1.1.3" | ||||
|  | ||||
| android-props-minSdk = "21" | ||||
| android-props-compileSdk = "33" | ||||
| android-props-buildTools = "33.0.0" | ||||
| android-props-buildTools = "33.0.1" | ||||
|  | ||||
| [libraries] | ||||
|  | ||||
|   | ||||
							
								
								
									
										2
									
								
								gradle/wrapper/gradle-wrapper.properties
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								gradle/wrapper/gradle-wrapper.properties
									
									
									
									
										vendored
									
									
								
							| @@ -1,5 +1,5 @@ | ||||
| distributionBase=GRADLE_USER_HOME | ||||
| distributionPath=wrapper/dists | ||||
| distributionUrl=https\://services.gradle.org/distributions/gradle-7.5.1-bin.zip | ||||
| distributionUrl=https\://services.gradle.org/distributions/gradle-7.6-bin.zip | ||||
| zipStoreBase=GRADLE_USER_HOME | ||||
| zipStorePath=wrapper/dists | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| package dev.inmo.micro_utils.koin | ||||
|  | ||||
| import org.koin.core.definition.Definition | ||||
| import org.koin.core.definition.KoinDefinition | ||||
| import org.koin.core.instance.InstanceFactory | ||||
| import org.koin.core.module.Module | ||||
| import org.koin.core.qualifier.Qualifier | ||||
| @@ -13,7 +14,7 @@ inline fun <reified T : Any> Module.factoryWithBinds( | ||||
|     qualifier: Qualifier? = null, | ||||
|     bindFilter: (KClass<*>) -> Boolean = { true }, | ||||
|     noinline definition: Definition<T> | ||||
| ): Pair<Module, InstanceFactory<*>> { | ||||
| ): KoinDefinition<*> { | ||||
|     return factory(qualifier, definition) binds (T::class.allSuperclasses.filter(bindFilter).toTypedArray()) | ||||
| } | ||||
|  | ||||
| @@ -21,7 +22,7 @@ inline fun <reified T : Any> Module.factoryWithBinds( | ||||
|     qualifier: String, | ||||
|     bindFilter: (KClass<*>) -> Boolean = { true }, | ||||
|     noinline definition: Definition<T> | ||||
| ): Pair<Module, InstanceFactory<*>> { | ||||
| ): KoinDefinition<*> { | ||||
|     return factory(qualifier, definition) binds (T::class.allSuperclasses.filter(bindFilter).toTypedArray()) | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| package dev.inmo.micro_utils.koin | ||||
|  | ||||
| import org.koin.core.definition.Definition | ||||
| import org.koin.core.definition.KoinDefinition | ||||
| import org.koin.core.instance.InstanceFactory | ||||
| import org.koin.core.module.Module | ||||
| import kotlin.reflect.KClass | ||||
| @@ -8,6 +9,6 @@ import kotlin.reflect.KClass | ||||
| inline fun <reified T : Any> Module.factoryWithRandomQualifierAndBinds( | ||||
|     bindFilter: (KClass<*>) -> Boolean = { true }, | ||||
|     noinline definition: Definition<T> | ||||
| ): Pair<Module, InstanceFactory<*>> { | ||||
| ): KoinDefinition<*> { | ||||
|     return factoryWithBinds(RandomQualifier(), bindFilter, definition) | ||||
| } | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| package dev.inmo.micro_utils.koin | ||||
|  | ||||
| import org.koin.core.definition.Definition | ||||
| import org.koin.core.definition.KoinDefinition | ||||
| import org.koin.core.instance.InstanceFactory | ||||
| import org.koin.core.module.Module | ||||
| import org.koin.core.qualifier.Qualifier | ||||
| @@ -14,7 +15,7 @@ inline fun <reified T : Any> Module.singleWithBinds( | ||||
|     createdAtStart: Boolean = false, | ||||
|     bindFilter: (KClass<*>) -> Boolean = { true }, | ||||
|     noinline definition: Definition<T> | ||||
| ): Pair<Module, InstanceFactory<*>> { | ||||
| ): KoinDefinition<*> { | ||||
|     return single(qualifier, createdAtStart, definition) binds (T::class.allSuperclasses.filter(bindFilter).toTypedArray()) | ||||
| } | ||||
|  | ||||
| @@ -24,7 +25,7 @@ inline fun <reified T : Any> Module.singleWithBinds( | ||||
|     createdAtStart: Boolean = false, | ||||
|     bindFilter: (KClass<*>) -> Boolean = { true }, | ||||
|     noinline definition: Definition<T> | ||||
| ): Pair<Module, InstanceFactory<*>> { | ||||
| ): KoinDefinition<*> { | ||||
|     return single(qualifier, createdAtStart, definition) binds (T::class.allSuperclasses.filter(bindFilter).toTypedArray()) | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| package dev.inmo.micro_utils.koin | ||||
|  | ||||
| import org.koin.core.definition.Definition | ||||
| import org.koin.core.definition.KoinDefinition | ||||
| import org.koin.core.instance.InstanceFactory | ||||
| import org.koin.core.module.Module | ||||
| import kotlin.reflect.KClass | ||||
| @@ -9,6 +10,6 @@ inline fun <reified T : Any> Module.singleWithRandomQualifierAndBinds( | ||||
|     createdAtStart: Boolean = false, | ||||
|     bindFilter: (KClass<*>) -> Boolean = { true }, | ||||
|     noinline definition: Definition<T> | ||||
| ): Pair<Module, InstanceFactory<*>> { | ||||
| ): KoinDefinition<*> { | ||||
|     return singleWithBinds(RandomQualifier(), createdAtStart, bindFilter, definition) | ||||
| } | ||||
|   | ||||
| @@ -4,6 +4,10 @@ import io.ktor.client.call.body | ||||
| import io.ktor.client.statement.HttpResponse | ||||
| import io.ktor.http.HttpStatusCode | ||||
|  | ||||
| suspend inline fun <reified T : Any> HttpResponse.bodyOrNull() = takeIf { | ||||
|     status == HttpStatusCode.OK | ||||
| } ?.body<T>() | ||||
| suspend inline fun <reified T : Any> HttpResponse.bodyOrNull( | ||||
|     statusFilter: (HttpResponse) -> Boolean = { it.status == HttpStatusCode.OK } | ||||
| ) = takeIf(statusFilter) ?.body<T>() | ||||
|  | ||||
| suspend inline fun <reified T : Any> HttpResponse.bodyOrNullOnNoContent() = bodyOrNull<T> { | ||||
|     it.status != HttpStatusCode.NoContent | ||||
| } | ||||
|   | ||||
| @@ -19,7 +19,7 @@ import kotlinx.coroutines.isActive | ||||
|  * connection. Must return true in case if must be reconnected. By default always reconnecting | ||||
|  */ | ||||
| @Warning("This feature is internal and should not be used directly. It is can be changed without any notification and warranty on compile-time or other guaranties") | ||||
| inline fun <reified T : Any> openBaseWebSocketFlow( | ||||
| inline fun <T : Any> openBaseWebSocketFlow( | ||||
|     noinline checkReconnection: suspend (Throwable?) -> Boolean = { true }, | ||||
|     noinline webSocketSessionRequest: suspend SendChannel<T>.() -> Unit | ||||
| ): Flow<T> { | ||||
| @@ -57,7 +57,7 @@ inline fun <reified T : Any> HttpClient.openWebSocketFlow( | ||||
| ): Flow<T> { | ||||
|     pluginOrNull(WebSockets) ?: error("Plugin $WebSockets must be installed for using createStandardWebsocketFlow") | ||||
|  | ||||
|     return openBaseWebSocketFlow<T>(checkReconnection) { | ||||
|     return openBaseWebSocketFlow(checkReconnection) { | ||||
|         val block: suspend DefaultClientWebSocketSession.() -> Unit = { | ||||
|             while (isActive) { | ||||
|                 send(receiveDeserialized<T>()) | ||||
|   | ||||
| @@ -17,8 +17,11 @@ import io.ktor.http.HttpStatusCode | ||||
| import io.ktor.http.Parameters | ||||
| import io.ktor.http.content.PartData | ||||
| import kotlinx.serialization.DeserializationStrategy | ||||
| import kotlinx.serialization.InternalSerializationApi | ||||
| import kotlinx.serialization.SerializationStrategy | ||||
| import kotlinx.serialization.StringFormat | ||||
| import kotlinx.serialization.encodeToString | ||||
| import kotlinx.serialization.serializer | ||||
| import java.io.File | ||||
|  | ||||
| /** | ||||
| @@ -29,6 +32,7 @@ import java.io.File | ||||
|  * in case you wish to pass other source of multipart binary data than regular file | ||||
|  * @suppress | ||||
|  */ | ||||
| @OptIn(InternalSerializationApi::class) | ||||
| actual suspend fun <T> HttpClient.uniUpload( | ||||
|     url: String, | ||||
|     data: Map<String, Any>, | ||||
| @@ -60,7 +64,7 @@ actual suspend fun <T> HttpClient.uniUpload( | ||||
|                 ) | ||||
|                 else -> append( | ||||
|                     k, | ||||
|                     stringFormat.encodeToString(v) | ||||
|                     stringFormat.encodeToString(v::class.serializer() as SerializationStrategy<in Any>, v) | ||||
|                 ) | ||||
|             } | ||||
|         } | ||||
|   | ||||
| @@ -10,8 +10,8 @@ import java.io.File | ||||
|  | ||||
| fun PartData.FileItem.download(target: File) { | ||||
|     provider().use { input -> | ||||
|         target.outputStream().use { | ||||
|             input.copyTo(it.asOutput()) | ||||
|         target.outputStream().asOutput().use { | ||||
|             input.copyTo(it) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -0,0 +1,29 @@ | ||||
| package dev.inmo.micro_utils.pagination.utils | ||||
|  | ||||
| fun <T> Iterable<T>.optionallyReverse(reverse: Boolean): Iterable<T> = when (this) { | ||||
|     is List<T> -> optionallyReverse(reverse) | ||||
|     is Set<T> -> optionallyReverse(reverse) | ||||
|     else -> if (reverse) { | ||||
|         reversed() | ||||
|     } else { | ||||
|         this | ||||
|     } | ||||
| } | ||||
| fun <T> List<T>.optionallyReverse(reverse: Boolean): List<T> = if (reverse) { | ||||
|     reversed() | ||||
| } else { | ||||
|     this | ||||
| } | ||||
| fun <T> Set<T>.optionallyReverse(reverse: Boolean): Set<T> = if (reverse) { | ||||
|     reversed().toSet() | ||||
| } else { | ||||
|     this | ||||
| } | ||||
|  | ||||
| inline fun <reified T> Array<T>.optionallyReverse(reverse: Boolean) = if (reverse) { | ||||
|     Array(size) { | ||||
|         get(lastIndex - it) | ||||
|     } | ||||
| } else { | ||||
|     this | ||||
| } | ||||
| @@ -32,6 +32,24 @@ fun <T> List<T>.paginate(with: Pagination): PaginationResult<T> { | ||||
|     ) | ||||
| } | ||||
| 
 | ||||
| fun <T> List<T>.paginate(with: Pagination, reversed: Boolean): PaginationResult<T> { | ||||
|     val actualPagination = with.optionallyReverse( | ||||
|         size, | ||||
|         reversed | ||||
|     ) | ||||
| 
 | ||||
|     val firstIndex = maxOf(actualPagination.firstIndex, 0) | ||||
|     val lastIndex = minOf(actualPagination.lastIndexExclusive, size) | ||||
|     if (firstIndex > lastIndex) { | ||||
|         return emptyPaginationResult() | ||||
|     } | ||||
| 
 | ||||
|     return subList(firstIndex, lastIndex).optionallyReverse(reversed).createPaginationResult( | ||||
|         with, | ||||
|         size.toLong() | ||||
|     ) | ||||
| } | ||||
| 
 | ||||
| fun <T> Set<T>.paginate(with: Pagination): PaginationResult<T> { | ||||
|     return this.drop(with.firstIndex).take(with.size).createPaginationResult( | ||||
|         with, | ||||
| @@ -39,30 +57,20 @@ fun <T> Set<T>.paginate(with: Pagination): PaginationResult<T> { | ||||
|     ) | ||||
| } | ||||
| 
 | ||||
| fun <T> Iterable<T>.optionallyReverse(reverse: Boolean): Iterable<T> = when (this) { | ||||
|     is List<T> -> optionallyReverse(reverse) | ||||
|     is Set<T> -> optionallyReverse(reverse) | ||||
|     else -> if (reverse) { | ||||
|         reversed() | ||||
|     } else { | ||||
|         this | ||||
|     } | ||||
| } | ||||
| fun <T> List<T>.optionallyReverse(reverse: Boolean): List<T> = if (reverse) { | ||||
|     reversed() | ||||
| } else { | ||||
|     this | ||||
| } | ||||
| fun <T> Set<T>.optionallyReverse(reverse: Boolean): Set<T> = if (reverse) { | ||||
|     reversed().toSet() | ||||
| } else { | ||||
|     this | ||||
| } | ||||
| fun <T> Set<T>.paginate(with: Pagination, reversed: Boolean): PaginationResult<T> { | ||||
|     val actualPagination = with.optionallyReverse( | ||||
|         size, | ||||
|         reversed | ||||
|     ) | ||||
| 
 | ||||
| inline fun <reified T> Array<T>.optionallyReverse(reverse: Boolean) = if (reverse) { | ||||
|     Array(size) { | ||||
|         get(lastIndex - it) | ||||
|     val firstIndex = maxOf(actualPagination.firstIndex, 0) | ||||
|     val lastIndex = minOf(actualPagination.lastIndexExclusive, size) | ||||
|     if (firstIndex > lastIndex) { | ||||
|         return emptyPaginationResult() | ||||
|     } | ||||
| } else { | ||||
|     this | ||||
| 
 | ||||
|     return this.drop(firstIndex).take(lastIndex - firstIndex).optionallyReverse(reversed).createPaginationResult( | ||||
|         with, | ||||
|         size.toLong() | ||||
|     ) | ||||
| } | ||||
| @@ -10,12 +10,14 @@ open class ReadCRUDCacheRepo<ObjectType, IdType>( | ||||
|     protected open val parentRepo: ReadCRUDRepo<ObjectType, IdType>, | ||||
|     protected open val kvCache: KVCache<IdType, ObjectType>, | ||||
|     protected open val idGetter: (ObjectType) -> IdType | ||||
| ) : ReadCRUDRepo<ObjectType, IdType> by parentRepo, CacheRepo { | ||||
| ) : ReadCRUDRepo<ObjectType, IdType> by parentRepo, CommonCacheRepo { | ||||
|     override suspend fun getById(id: IdType): ObjectType? = kvCache.get(id) ?: (parentRepo.getById(id) ?.also { | ||||
|         kvCache.set(id, it) | ||||
|     }) | ||||
|  | ||||
|     override suspend fun contains(id: IdType): Boolean = kvCache.contains(id) || parentRepo.contains(id) | ||||
|  | ||||
|     override suspend fun invalidate() = kvCache.clear() | ||||
| } | ||||
|  | ||||
| fun <ObjectType, IdType> ReadCRUDRepo<ObjectType, IdType>.cached( | ||||
| @@ -28,7 +30,7 @@ open class WriteCRUDCacheRepo<ObjectType, IdType, InputValueType>( | ||||
|     protected open val kvCache: KVCache<IdType, ObjectType>, | ||||
|     protected open val scope: CoroutineScope = CoroutineScope(Dispatchers.Default), | ||||
|     protected open val idGetter: (ObjectType) -> IdType | ||||
| ) : WriteCRUDRepo<ObjectType, IdType, InputValueType>, CacheRepo { | ||||
| ) : WriteCRUDRepo<ObjectType, IdType, InputValueType>, CommonCacheRepo { | ||||
|     override val newObjectsFlow: Flow<ObjectType> by parentRepo::newObjectsFlow | ||||
|     override val updatedObjectsFlow: Flow<ObjectType> by parentRepo::updatedObjectsFlow | ||||
|     override val deletedObjectsIdsFlow: Flow<IdType> by parentRepo::deletedObjectsIdsFlow | ||||
| @@ -72,6 +74,8 @@ open class WriteCRUDCacheRepo<ObjectType, IdType, InputValueType>( | ||||
|  | ||||
|         return created | ||||
|     } | ||||
|  | ||||
|     override suspend fun invalidate() = kvCache.clear() | ||||
| } | ||||
|  | ||||
| fun <ObjectType, IdType, InputType> WriteCRUDRepo<ObjectType, IdType, InputType>.caching( | ||||
|   | ||||
| @@ -1,3 +1,5 @@ | ||||
| package dev.inmo.micro_utils.repos.cache | ||||
|  | ||||
| interface CacheRepo | ||||
| interface CacheRepo { | ||||
|     suspend fun invalidate() | ||||
| } | ||||
|   | ||||
							
								
								
									
										7
									
								
								repos/cache/src/commonMain/kotlin/dev/inmo/micro_utils/repos/cache/CommonCacheRepo.kt
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								repos/cache/src/commonMain/kotlin/dev/inmo/micro_utils/repos/cache/CommonCacheRepo.kt
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | ||||
| package dev.inmo.micro_utils.repos.cache | ||||
|  | ||||
| /** | ||||
|  * Any inheritor of this should work with next logic: try to take data from some [dev.inmo.micro_utils.repos.cache.cache.KVCache] and, | ||||
|  * if not exists, take from origin and save to the cache for future reuse | ||||
|  */ | ||||
| interface CommonCacheRepo : CacheRepo | ||||
							
								
								
									
										7
									
								
								repos/cache/src/commonMain/kotlin/dev/inmo/micro_utils/repos/cache/FallbackCacheRepo.kt
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								repos/cache/src/commonMain/kotlin/dev/inmo/micro_utils/repos/cache/FallbackCacheRepo.kt
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | ||||
| package dev.inmo.micro_utils.repos.cache | ||||
|  | ||||
| /** | ||||
|  * Any inheritor of this should work with next logic: try to take data from their original repo, if successful - save data to internal | ||||
|  * [dev.inmo.micro_utils.repos.cache.cache.FullKVCache] or try to take data from that internal cache | ||||
|  */ | ||||
| interface FallbackCacheRepo : CacheRepo | ||||
| @@ -10,7 +10,7 @@ import kotlinx.coroutines.flow.* | ||||
| open class ReadKeyValueCacheRepo<Key,Value>( | ||||
|     protected open val parentRepo: ReadKeyValueRepo<Key, Value>, | ||||
|     protected open val kvCache: KVCache<Key, Value>, | ||||
| ) : ReadKeyValueRepo<Key,Value> by parentRepo, CacheRepo { | ||||
| ) : ReadKeyValueRepo<Key,Value> by parentRepo, CommonCacheRepo { | ||||
|     override suspend fun get(k: Key): Value? = kvCache.get(k) ?: parentRepo.get(k) ?.also { kvCache.set(k, it) } | ||||
|     override suspend fun contains(key: Key): Boolean = kvCache.contains(key) || parentRepo.contains(key) | ||||
|  | ||||
| @@ -23,6 +23,8 @@ open class ReadKeyValueCacheRepo<Key,Value>( | ||||
|             ) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     override suspend fun invalidate() = kvCache.clear() | ||||
| } | ||||
|  | ||||
| fun <Key, Value> ReadKeyValueRepo<Key, Value>.cached( | ||||
| @@ -33,9 +35,11 @@ open class KeyValueCacheRepo<Key,Value>( | ||||
|     parentRepo: KeyValueRepo<Key, Value>, | ||||
|     kvCache: KVCache<Key, Value>, | ||||
|     scope: CoroutineScope = CoroutineScope(Dispatchers.Default) | ||||
| ) : ReadKeyValueCacheRepo<Key,Value>(parentRepo, kvCache), KeyValueRepo<Key,Value>, WriteKeyValueRepo<Key, Value> by parentRepo, CacheRepo { | ||||
| ) : ReadKeyValueCacheRepo<Key,Value>(parentRepo, kvCache), KeyValueRepo<Key,Value>, WriteKeyValueRepo<Key, Value> by parentRepo, CommonCacheRepo { | ||||
|     protected val onNewJob = parentRepo.onNewValue.onEach { kvCache.set(it.first, it.second) }.launchIn(scope) | ||||
|     protected val onRemoveJob = parentRepo.onValueRemoved.onEach { kvCache.unset(it) }.launchIn(scope) | ||||
|  | ||||
|     override suspend fun invalidate() = kvCache.clear() | ||||
| } | ||||
|  | ||||
| fun <Key, Value> KeyValueRepo<Key, Value>.cached( | ||||
|   | ||||
| @@ -11,7 +11,7 @@ import kotlinx.coroutines.flow.* | ||||
| open class ReadKeyValuesCacheRepo<Key,Value>( | ||||
|     protected open val parentRepo: ReadKeyValuesRepo<Key, Value>, | ||||
|     protected open val kvCache: KVCache<Key, List<Value>> | ||||
| ) : ReadKeyValuesRepo<Key,Value> by parentRepo, CacheRepo { | ||||
| ) : ReadKeyValuesRepo<Key,Value> by parentRepo, CommonCacheRepo { | ||||
|     override suspend fun get(k: Key, pagination: Pagination, reversed: Boolean): PaginationResult<Value> { | ||||
|         return getAll(k, reversed).paginate( | ||||
|             pagination | ||||
| @@ -30,6 +30,8 @@ open class ReadKeyValuesCacheRepo<Key,Value>( | ||||
|         } | ||||
|     }) | ||||
|     override suspend fun contains(k: Key): Boolean = kvCache.contains(k) || parentRepo.contains(k) | ||||
|  | ||||
|     override suspend fun invalidate() = kvCache.clear() | ||||
| } | ||||
|  | ||||
| fun <Key, Value> ReadKeyValuesRepo<Key, Value>.cached( | ||||
| @@ -40,7 +42,7 @@ open class KeyValuesCacheRepo<Key,Value>( | ||||
|     parentRepo: KeyValuesRepo<Key, Value>, | ||||
|     kvCache: KVCache<Key, List<Value>>, | ||||
|     scope: CoroutineScope = CoroutineScope(Dispatchers.Default) | ||||
| ) : ReadKeyValuesCacheRepo<Key,Value>(parentRepo, kvCache), KeyValuesRepo<Key,Value>, WriteKeyValuesRepo<Key,Value> by parentRepo, CacheRepo { | ||||
| ) : ReadKeyValuesCacheRepo<Key,Value>(parentRepo, kvCache), KeyValuesRepo<Key,Value>, WriteKeyValuesRepo<Key,Value> by parentRepo, CommonCacheRepo { | ||||
|     protected val onNewJob = parentRepo.onNewValue.onEach { (k, v) -> | ||||
|         kvCache.set( | ||||
|             k, | ||||
| @@ -56,6 +58,8 @@ open class KeyValuesCacheRepo<Key,Value>( | ||||
|     protected val onDataClearedJob = parentRepo.onDataCleared.onEach { | ||||
|         kvCache.unset(it) | ||||
|     }.launchIn(scope) | ||||
|  | ||||
|     override suspend fun invalidate() = kvCache.clear() | ||||
| } | ||||
|  | ||||
| fun <Key, Value> KeyValuesRepo<Key, Value>.cached( | ||||
|   | ||||
							
								
								
									
										37
									
								
								repos/cache/src/commonMain/kotlin/dev/inmo/micro_utils/repos/cache/fallback/ActionWrapper.kt
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								repos/cache/src/commonMain/kotlin/dev/inmo/micro_utils/repos/cache/fallback/ActionWrapper.kt
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,37 @@ | ||||
| package dev.inmo.micro_utils.repos.cache.fallback | ||||
|  | ||||
| import dev.inmo.micro_utils.coroutines.runCatchingSafely | ||||
| import kotlinx.coroutines.withTimeout | ||||
|  | ||||
| /** | ||||
|  * Realizations should [wrap] the work with some conditions like retries on exceptions, calling timeout, etc. | ||||
|  *  | ||||
|  * @see Timeouted | ||||
|  * @see Direct | ||||
|  */ | ||||
| interface ActionWrapper { | ||||
|     /** | ||||
|      * Should execute [block] to take the result [T], but may return failure in case when something went wrong. | ||||
|      * This method should never throw any [Exception] | ||||
|      */ | ||||
|     suspend fun <T> wrap(block: suspend () -> T): Result<T> | ||||
|  | ||||
|     /** | ||||
|      * This type of [ActionWrapper]s will use [withTimeout]([timeoutMillis]) and if original call | ||||
|      * will not return anything in that timeout just return [Result] with failure | ||||
|      */ | ||||
|     class Timeouted(private val timeoutMillis: Long) : ActionWrapper { | ||||
|         override suspend fun <T> wrap(block: suspend () -> T): Result<T> = runCatchingSafely { | ||||
|             withTimeout(timeoutMillis) { | ||||
|                 block() | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|     /** | ||||
|      * It is passthrough variant of [ActionWrapper] which will just call incoming block with wrapping into [runCatchingSafely] | ||||
|      */ | ||||
|     object Direct : ActionWrapper { | ||||
|  | ||||
|         override suspend fun <T> wrap(block: suspend () -> T): Result<T> = runCatchingSafely { block() } | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,36 @@ | ||||
| package dev.inmo.micro_utils.repos.cache.fallback.crud | ||||
|  | ||||
| import dev.inmo.micro_utils.repos.CRUDRepo | ||||
| import dev.inmo.micro_utils.repos.WriteCRUDRepo | ||||
| import dev.inmo.micro_utils.repos.cache.cache.FullKVCache | ||||
| import dev.inmo.micro_utils.repos.cache.fallback.ActionWrapper | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlin.time.Duration.Companion.seconds | ||||
|  | ||||
| open class AutoRecacheCRUDRepo<RegisteredObject, Id, InputObject>( | ||||
|     originalRepo: CRUDRepo<RegisteredObject, Id, InputObject>, | ||||
|     scope: CoroutineScope, | ||||
|     kvCache: FullKVCache<Id, RegisteredObject> = FullKVCache(), | ||||
|     recacheDelay: Long = 60.seconds.inWholeMilliseconds, | ||||
|     actionWrapper: ActionWrapper = ActionWrapper.Direct, | ||||
|     idGetter: (RegisteredObject) -> Id | ||||
| ) : AutoRecacheReadCRUDRepo<RegisteredObject, Id>( | ||||
|     originalRepo, | ||||
|     scope, | ||||
|     kvCache, | ||||
|     recacheDelay, | ||||
|     actionWrapper, | ||||
|     idGetter | ||||
| ), | ||||
|     WriteCRUDRepo<RegisteredObject, Id, InputObject> by AutoRecacheWriteCRUDRepo(originalRepo, scope, kvCache, idGetter), | ||||
|     CRUDRepo<RegisteredObject, Id, InputObject> { | ||||
|  | ||||
|     constructor( | ||||
|         originalRepo: CRUDRepo<RegisteredObject, Id, InputObject>, | ||||
|         scope: CoroutineScope, | ||||
|         originalCallTimeoutMillis: Long, | ||||
|         kvCache: FullKVCache<Id, RegisteredObject> = FullKVCache(), | ||||
|         recacheDelay: Long = 60.seconds.inWholeMilliseconds, | ||||
|         idGetter: (RegisteredObject) -> Id | ||||
|     ) : this(originalRepo, scope, kvCache, recacheDelay, ActionWrapper.Timeouted(originalCallTimeoutMillis), idGetter) | ||||
| } | ||||
| @@ -0,0 +1,86 @@ | ||||
| package dev.inmo.micro_utils.repos.cache.fallback.crud | ||||
|  | ||||
| import dev.inmo.micro_utils.coroutines.runCatchingSafely | ||||
| import dev.inmo.micro_utils.pagination.Pagination | ||||
| import dev.inmo.micro_utils.pagination.PaginationResult | ||||
| import dev.inmo.micro_utils.repos.ReadCRUDRepo | ||||
| import dev.inmo.micro_utils.repos.cache.cache.FullKVCache | ||||
| import dev.inmo.micro_utils.repos.cache.fallback.ActionWrapper | ||||
| import dev.inmo.micro_utils.repos.cache.util.actualizeAll | ||||
| import dev.inmo.micro_utils.repos.cache.FallbackCacheRepo | ||||
| import dev.inmo.micro_utils.repos.set | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlinx.coroutines.delay | ||||
| import kotlinx.coroutines.isActive | ||||
| import kotlinx.coroutines.launch | ||||
| import kotlin.time.Duration.Companion.seconds | ||||
|  | ||||
| open class AutoRecacheReadCRUDRepo<RegisteredObject, Id>( | ||||
|     protected open val originalRepo: ReadCRUDRepo<RegisteredObject, Id>, | ||||
|     protected val scope: CoroutineScope, | ||||
|     protected val kvCache: FullKVCache<Id, RegisteredObject> = FullKVCache(), | ||||
|     protected val recacheDelay: Long = 60.seconds.inWholeMilliseconds, | ||||
|     protected val actionWrapper: ActionWrapper = ActionWrapper.Direct, | ||||
|     protected val idGetter: (RegisteredObject) -> Id | ||||
| ) : ReadCRUDRepo<RegisteredObject, Id>, FallbackCacheRepo { | ||||
|     val autoUpdateJob = scope.launch { | ||||
|         while (isActive) { | ||||
|             actualizeAll() | ||||
|  | ||||
|             delay(recacheDelay) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     constructor( | ||||
|         originalRepo: ReadCRUDRepo<RegisteredObject, Id>, | ||||
|         scope: CoroutineScope, | ||||
|         originalCallTimeoutMillis: Long, | ||||
|         kvCache: FullKVCache<Id, RegisteredObject> = FullKVCache(), | ||||
|         recacheDelay: Long = 60.seconds.inWholeMilliseconds, | ||||
|         idGetter: (RegisteredObject) -> Id | ||||
|     ) : this(originalRepo, scope, kvCache, recacheDelay, ActionWrapper.Timeouted(originalCallTimeoutMillis), idGetter) | ||||
|  | ||||
|     protected open suspend fun actualizeAll(): Result<Unit> { | ||||
|         return runCatchingSafely { | ||||
|             kvCache.actualizeAll(originalRepo) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     override suspend fun contains(id: Id): Boolean = actionWrapper.wrap { | ||||
|         originalRepo.contains(id) | ||||
|     }.getOrElse { | ||||
|         kvCache.contains(id) | ||||
|     } | ||||
|  | ||||
|     override suspend fun count(): Long = actionWrapper.wrap { | ||||
|         originalRepo.count() | ||||
|     }.getOrElse { | ||||
|         kvCache.count() | ||||
|     } | ||||
|  | ||||
|     override suspend fun getByPagination( | ||||
|         pagination: Pagination | ||||
|     ): PaginationResult<RegisteredObject> = actionWrapper.wrap { | ||||
|         originalRepo.getByPagination(pagination) | ||||
|     }.getOrNull() ?.also { | ||||
|         it.results.forEach { | ||||
|             kvCache.set(idGetter(it), it) | ||||
|         } | ||||
|     } ?: kvCache.values(pagination) | ||||
|  | ||||
|     override suspend fun getIdsByPagination( | ||||
|         pagination: Pagination | ||||
|     ): PaginationResult<Id> = actionWrapper.wrap { | ||||
|         originalRepo.getIdsByPagination(pagination) | ||||
|     }.getOrElse { kvCache.keys(pagination) } | ||||
|  | ||||
|     override suspend fun getById(id: Id): RegisteredObject? = actionWrapper.wrap { | ||||
|         originalRepo.getById(id) | ||||
|     }.getOrNull() ?.also { | ||||
|         kvCache.set(idGetter(it), it) | ||||
|     } ?: kvCache.get(id) | ||||
|  | ||||
|     override suspend fun invalidate() { | ||||
|         actualizeAll() | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,65 @@ | ||||
| package dev.inmo.micro_utils.repos.cache.fallback.crud | ||||
|  | ||||
| import dev.inmo.micro_utils.coroutines.plus | ||||
| import dev.inmo.micro_utils.coroutines.subscribeSafelyWithoutExceptions | ||||
| import dev.inmo.micro_utils.repos.UpdatedValuePair | ||||
| import dev.inmo.micro_utils.repos.WriteCRUDRepo | ||||
| import dev.inmo.micro_utils.repos.cache.cache.FullKVCache | ||||
| import dev.inmo.micro_utils.repos.cache.FallbackCacheRepo | ||||
| import dev.inmo.micro_utils.repos.set | ||||
| import dev.inmo.micro_utils.repos.unset | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlinx.coroutines.flow.Flow | ||||
| import kotlinx.coroutines.flow.distinctUntilChanged | ||||
| import kotlinx.coroutines.flow.map | ||||
| import kotlinx.coroutines.flow.merge | ||||
|  | ||||
| open class AutoRecacheWriteCRUDRepo<RegisteredObject, Id, InputObject>( | ||||
|     protected val originalRepo: WriteCRUDRepo<RegisteredObject, Id, InputObject>, | ||||
|     protected val scope: CoroutineScope, | ||||
|     protected val kvCache: FullKVCache<Id, RegisteredObject> = FullKVCache(), | ||||
|     protected val idGetter: (RegisteredObject) -> Id | ||||
| ) : WriteCRUDRepo<RegisteredObject, Id, InputObject>, FallbackCacheRepo { | ||||
|     override val deletedObjectsIdsFlow: Flow<Id> | ||||
|         get() = (originalRepo.deletedObjectsIdsFlow + kvCache.onValueRemoved).distinctUntilChanged() | ||||
|     override val newObjectsFlow: Flow<RegisteredObject> | ||||
|         get() = (originalRepo.newObjectsFlow + kvCache.onNewValue.map { it.second }).distinctUntilChanged() | ||||
|     override val updatedObjectsFlow: Flow<RegisteredObject> | ||||
|         get() = originalRepo.updatedObjectsFlow | ||||
|  | ||||
|     private val onRemovingUpdatesListeningJob = originalRepo.deletedObjectsIdsFlow.subscribeSafelyWithoutExceptions(scope) { | ||||
|         kvCache.unset(it) | ||||
|     } | ||||
|  | ||||
|     private val onNewAndUpdatedObjectsListeningJob = merge( | ||||
|         originalRepo.newObjectsFlow, | ||||
|         originalRepo.updatedObjectsFlow, | ||||
|     ).subscribeSafelyWithoutExceptions(scope) { | ||||
|         kvCache.set(idGetter(it), it) | ||||
|     } | ||||
|  | ||||
|     override suspend fun update( | ||||
|         values: List<UpdatedValuePair<Id, InputObject>> | ||||
|     ): List<RegisteredObject> = originalRepo.update(values).onEach { | ||||
|         kvCache.set(idGetter(it), it) | ||||
|     } | ||||
|  | ||||
|     override suspend fun update( | ||||
|         id: Id, | ||||
|         value: InputObject | ||||
|     ): RegisteredObject? = originalRepo.update(id, value) ?.also { | ||||
|         kvCache.set(idGetter(it), it) | ||||
|     } | ||||
|  | ||||
|     override suspend fun deleteById(ids: List<Id>) = originalRepo.deleteById(ids).also { | ||||
|         kvCache.unset(ids) | ||||
|     } | ||||
|  | ||||
|     override suspend fun create(values: List<InputObject>): List<RegisteredObject> = originalRepo.create(values).onEach { | ||||
|         kvCache.set(idGetter(it), it) | ||||
|     } | ||||
|  | ||||
|     override suspend fun invalidate() { | ||||
|         kvCache.clear() | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,42 @@ | ||||
| package dev.inmo.micro_utils.repos.cache.fallback.keyvalue | ||||
|  | ||||
| import dev.inmo.micro_utils.repos.KeyValueRepo | ||||
| import dev.inmo.micro_utils.repos.WriteKeyValueRepo | ||||
| import dev.inmo.micro_utils.repos.cache.cache.FullKVCache | ||||
| import dev.inmo.micro_utils.repos.cache.fallback.ActionWrapper | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlin.time.Duration.Companion.seconds | ||||
|  | ||||
| open class AutoRecacheKeyValueRepo<Id, RegisteredObject>( | ||||
|     override val originalRepo: KeyValueRepo<Id, RegisteredObject>, | ||||
|     scope: CoroutineScope, | ||||
|     kvCache: FullKVCache<Id, RegisteredObject> = FullKVCache(), | ||||
|     recacheDelay: Long = 60.seconds.inWholeMilliseconds, | ||||
|     actionWrapper: ActionWrapper = ActionWrapper.Direct, | ||||
|     idGetter: (RegisteredObject) -> Id | ||||
| ) : AutoRecacheReadKeyValueRepo<Id, RegisteredObject> ( | ||||
|     originalRepo, | ||||
|     scope, | ||||
|     kvCache, | ||||
|     recacheDelay, | ||||
|     actionWrapper, | ||||
|     idGetter | ||||
| ), | ||||
|     WriteKeyValueRepo<Id, RegisteredObject> by AutoRecacheWriteKeyValueRepo(originalRepo, scope, kvCache), | ||||
|     KeyValueRepo<Id, RegisteredObject> { | ||||
|  | ||||
|     constructor( | ||||
|         originalRepo: KeyValueRepo<Id, RegisteredObject>, | ||||
|         scope: CoroutineScope, | ||||
|         originalCallTimeoutMillis: Long, | ||||
|         kvCache: FullKVCache<Id, RegisteredObject> = FullKVCache(), | ||||
|         recacheDelay: Long = 60.seconds.inWholeMilliseconds, | ||||
|         idGetter: (RegisteredObject) -> Id | ||||
|     ) : this(originalRepo, scope, kvCache, recacheDelay, ActionWrapper.Timeouted(originalCallTimeoutMillis), idGetter) | ||||
|  | ||||
|     override suspend fun unsetWithValues(toUnset: List<RegisteredObject>) = originalRepo.unsetWithValues( | ||||
|         toUnset | ||||
|     ).also { | ||||
|         kvCache.unsetWithValues(toUnset) | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,96 @@ | ||||
| package dev.inmo.micro_utils.repos.cache.fallback.keyvalue | ||||
|  | ||||
| import dev.inmo.micro_utils.coroutines.runCatchingSafely | ||||
| import dev.inmo.micro_utils.pagination.Pagination | ||||
| import dev.inmo.micro_utils.pagination.PaginationResult | ||||
| import dev.inmo.micro_utils.repos.ReadKeyValueRepo | ||||
| import dev.inmo.micro_utils.repos.cache.cache.FullKVCache | ||||
| import dev.inmo.micro_utils.repos.cache.fallback.ActionWrapper | ||||
| import dev.inmo.micro_utils.repos.cache.util.actualizeAll | ||||
| import dev.inmo.micro_utils.repos.cache.FallbackCacheRepo | ||||
| import dev.inmo.micro_utils.repos.set | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlinx.coroutines.delay | ||||
| import kotlinx.coroutines.isActive | ||||
| import kotlinx.coroutines.launch | ||||
| import kotlin.time.Duration.Companion.seconds | ||||
|  | ||||
| open class AutoRecacheReadKeyValueRepo<Id, RegisteredObject>( | ||||
|     protected open val originalRepo: ReadKeyValueRepo<Id, RegisteredObject>, | ||||
|     protected val scope: CoroutineScope, | ||||
|     protected val kvCache: FullKVCache<Id, RegisteredObject> = FullKVCache(), | ||||
|     protected val recacheDelay: Long = 60.seconds.inWholeMilliseconds, | ||||
|     protected val actionWrapper: ActionWrapper = ActionWrapper.Direct, | ||||
|     protected val idGetter: (RegisteredObject) -> Id | ||||
| ) : ReadKeyValueRepo<Id, RegisteredObject>, FallbackCacheRepo { | ||||
|     val autoUpdateJob = scope.launch { | ||||
|         while (isActive) { | ||||
|             actualizeAll() | ||||
|  | ||||
|             delay(recacheDelay) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     constructor( | ||||
|         originalRepo: ReadKeyValueRepo<Id, RegisteredObject>, | ||||
|         scope: CoroutineScope, | ||||
|         originalCallTimeoutMillis: Long, | ||||
|         kvCache: FullKVCache<Id, RegisteredObject> = FullKVCache(), | ||||
|         recacheDelay: Long = 60.seconds.inWholeMilliseconds, | ||||
|         idGetter: (RegisteredObject) -> Id | ||||
|     ) : this(originalRepo, scope, kvCache, recacheDelay, ActionWrapper.Timeouted(originalCallTimeoutMillis), idGetter) | ||||
|  | ||||
|     protected open suspend fun actualizeAll(): Result<Unit> { | ||||
|         return runCatchingSafely { | ||||
|             kvCache.actualizeAll(originalRepo) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     override suspend fun contains(key: Id): Boolean = actionWrapper.wrap { | ||||
|         originalRepo.contains(key) | ||||
|     }.getOrElse { | ||||
|         kvCache.contains(key) | ||||
|     } | ||||
|  | ||||
|     override suspend fun count(): Long = actionWrapper.wrap { | ||||
|         originalRepo.count() | ||||
|     }.getOrElse { | ||||
|         kvCache.count() | ||||
|     } | ||||
|  | ||||
|     override suspend fun get(k: Id): RegisteredObject? = actionWrapper.wrap { | ||||
|         originalRepo.get(k) | ||||
|     }.getOrNull() ?.also { | ||||
|         kvCache.set(k, it) | ||||
|     } ?: kvCache.get(k) | ||||
|  | ||||
|     override suspend fun values( | ||||
|         pagination: Pagination, | ||||
|         reversed: Boolean | ||||
|     ): PaginationResult<RegisteredObject> = actionWrapper.wrap { | ||||
|         originalRepo.values(pagination, reversed) | ||||
|     }.getOrNull() ?.also { | ||||
|         it.results.forEach { | ||||
|             kvCache.set(idGetter(it), it) | ||||
|         } | ||||
|     } ?: kvCache.values(pagination, reversed) | ||||
|  | ||||
|     override suspend fun keys( | ||||
|         pagination: Pagination, | ||||
|         reversed: Boolean | ||||
|     ): PaginationResult<Id> = actionWrapper.wrap { | ||||
|         originalRepo.keys(pagination, reversed) | ||||
|     }.getOrElse { kvCache.keys(pagination, reversed) } | ||||
|  | ||||
|     override suspend fun keys( | ||||
|         v: RegisteredObject, | ||||
|         pagination: Pagination, | ||||
|         reversed: Boolean | ||||
|     ): PaginationResult<Id> = actionWrapper.wrap { | ||||
|         originalRepo.keys(v, pagination, reversed) | ||||
|     }.getOrElse { kvCache.keys(v, pagination, reversed) } | ||||
|  | ||||
|     override suspend fun invalidate() { | ||||
|         actualizeAll() | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,54 @@ | ||||
| package dev.inmo.micro_utils.repos.cache.fallback.keyvalue | ||||
|  | ||||
| import dev.inmo.micro_utils.coroutines.plus | ||||
| import dev.inmo.micro_utils.coroutines.subscribeSafelyWithoutExceptions | ||||
| import dev.inmo.micro_utils.repos.WriteKeyValueRepo | ||||
| import dev.inmo.micro_utils.repos.cache.cache.FullKVCache | ||||
| import dev.inmo.micro_utils.repos.cache.FallbackCacheRepo | ||||
| import dev.inmo.micro_utils.repos.set | ||||
| import dev.inmo.micro_utils.repos.unset | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlinx.coroutines.flow.Flow | ||||
| import kotlinx.coroutines.flow.distinctUntilChanged | ||||
|  | ||||
| open class AutoRecacheWriteKeyValueRepo<Id, RegisteredObject>( | ||||
|     protected val originalRepo: WriteKeyValueRepo<Id, RegisteredObject>, | ||||
|     protected val scope: CoroutineScope, | ||||
|     protected val kvCache: FullKVCache<Id, RegisteredObject> = FullKVCache() | ||||
| ) : WriteKeyValueRepo<Id, RegisteredObject>, FallbackCacheRepo { | ||||
|     override val onValueRemoved: Flow<Id> | ||||
|         get() = (originalRepo.onValueRemoved + kvCache.onValueRemoved).distinctUntilChanged() | ||||
|  | ||||
|     override val onNewValue: Flow<Pair<Id, RegisteredObject>> | ||||
|         get() = (originalRepo.onNewValue + kvCache.onNewValue).distinctUntilChanged() | ||||
|  | ||||
|     private val onRemovingUpdatesListeningJob = originalRepo.onValueRemoved.subscribeSafelyWithoutExceptions(scope) { | ||||
|         kvCache.unset(it) | ||||
|     } | ||||
|  | ||||
|     private val onNewAndUpdatedObjectsListeningJob = originalRepo.onNewValue.subscribeSafelyWithoutExceptions(scope) { | ||||
|         kvCache.set(it.first, it.second) | ||||
|     } | ||||
|  | ||||
|     override suspend fun unsetWithValues(toUnset: List<RegisteredObject>) = originalRepo.unsetWithValues( | ||||
|         toUnset | ||||
|     ).also { | ||||
|         kvCache.unsetWithValues(toUnset) | ||||
|     } | ||||
|  | ||||
|     override suspend fun unset(toUnset: List<Id>) = originalRepo.unset( | ||||
|         toUnset | ||||
|     ).also { | ||||
|         kvCache.unset(toUnset) | ||||
|     } | ||||
|  | ||||
|     override suspend fun set(toSet: Map<Id, RegisteredObject>) = originalRepo.set( | ||||
|         toSet | ||||
|     ).also { | ||||
|         kvCache.set(toSet) | ||||
|     } | ||||
|  | ||||
|     override suspend fun invalidate() { | ||||
|         kvCache.clear() | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,37 @@ | ||||
| package dev.inmo.micro_utils.repos.cache.fallback.keyvalues | ||||
|  | ||||
| import dev.inmo.micro_utils.repos.KeyValuesRepo | ||||
| import dev.inmo.micro_utils.repos.WriteKeyValuesRepo | ||||
| import dev.inmo.micro_utils.repos.cache.cache.FullKVCache | ||||
| import dev.inmo.micro_utils.repos.cache.fallback.ActionWrapper | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlin.time.Duration.Companion.seconds | ||||
|  | ||||
| open class AutoRecacheKeyValuesRepo<Id, RegisteredObject>( | ||||
|     override val originalRepo: KeyValuesRepo<Id, RegisteredObject>, | ||||
|     scope: CoroutineScope, | ||||
|     kvCache: FullKVCache<Id, List<RegisteredObject>> = FullKVCache(), | ||||
|     recacheDelay: Long = 60.seconds.inWholeMilliseconds, | ||||
|     actionWrapper: ActionWrapper = ActionWrapper.Direct | ||||
| ) : AutoRecacheReadKeyValuesRepo<Id, RegisteredObject> ( | ||||
|     originalRepo, | ||||
|     scope, | ||||
|     kvCache, | ||||
|     recacheDelay, | ||||
|     actionWrapper | ||||
| ), | ||||
|     WriteKeyValuesRepo<Id, RegisteredObject> by AutoRecacheWriteKeyValuesRepo(originalRepo, scope, kvCache), | ||||
|     KeyValuesRepo<Id, RegisteredObject> { | ||||
|  | ||||
|     constructor( | ||||
|         originalRepo: KeyValuesRepo<Id, RegisteredObject>, | ||||
|         scope: CoroutineScope, | ||||
|         originalCallTimeoutMillis: Long, | ||||
|         kvCache: FullKVCache<Id, List<RegisteredObject>> = FullKVCache(), | ||||
|         recacheDelay: Long = 60.seconds.inWholeMilliseconds | ||||
|     ) : this(originalRepo, scope, kvCache, recacheDelay, ActionWrapper.Timeouted(originalCallTimeoutMillis)) | ||||
|  | ||||
|     override suspend fun clearWithValue(v: RegisteredObject) { | ||||
|         super.clearWithValue(v) | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,145 @@ | ||||
| package dev.inmo.micro_utils.repos.cache.fallback.keyvalues | ||||
|  | ||||
| import dev.inmo.micro_utils.coroutines.runCatchingSafely | ||||
| import dev.inmo.micro_utils.pagination.Pagination | ||||
| import dev.inmo.micro_utils.pagination.PaginationResult | ||||
| import dev.inmo.micro_utils.pagination.changeResultsUnchecked | ||||
| import dev.inmo.micro_utils.pagination.createPaginationResult | ||||
| import dev.inmo.micro_utils.pagination.emptyPaginationResult | ||||
| import dev.inmo.micro_utils.pagination.firstIndex | ||||
| import dev.inmo.micro_utils.pagination.utils.doForAllWithNextPaging | ||||
| import dev.inmo.micro_utils.pagination.utils.optionallyReverse | ||||
| import dev.inmo.micro_utils.pagination.utils.paginate | ||||
| import dev.inmo.micro_utils.repos.ReadKeyValuesRepo | ||||
| import dev.inmo.micro_utils.repos.cache.cache.FullKVCache | ||||
| import dev.inmo.micro_utils.repos.cache.fallback.ActionWrapper | ||||
| import dev.inmo.micro_utils.repos.cache.util.actualizeAll | ||||
| import dev.inmo.micro_utils.repos.cache.FallbackCacheRepo | ||||
| import dev.inmo.micro_utils.repos.set | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlinx.coroutines.delay | ||||
| import kotlinx.coroutines.isActive | ||||
| import kotlinx.coroutines.launch | ||||
| import kotlin.time.Duration.Companion.seconds | ||||
|  | ||||
| open class AutoRecacheReadKeyValuesRepo<Id, RegisteredObject>( | ||||
|     protected open val originalRepo: ReadKeyValuesRepo<Id, RegisteredObject>, | ||||
|     protected val scope: CoroutineScope, | ||||
|     protected val kvCache: FullKVCache<Id, List<RegisteredObject>> = FullKVCache(), | ||||
|     protected val recacheDelay: Long = 60.seconds.inWholeMilliseconds, | ||||
|     protected val actionWrapper: ActionWrapper = ActionWrapper.Direct | ||||
| ) : ReadKeyValuesRepo<Id, RegisteredObject>, FallbackCacheRepo { | ||||
|     val autoUpdateJob = scope.launch { | ||||
|         while (isActive) { | ||||
|             actualizeAll() | ||||
|  | ||||
|             delay(recacheDelay) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     constructor( | ||||
|         originalRepo: ReadKeyValuesRepo<Id, RegisteredObject>, | ||||
|         scope: CoroutineScope, | ||||
|         originalCallTimeoutMillis: Long, | ||||
|         kvCache: FullKVCache<Id, List<RegisteredObject>> = FullKVCache(), | ||||
|         recacheDelay: Long = 60.seconds.inWholeMilliseconds | ||||
|     ) : this(originalRepo, scope, kvCache, recacheDelay, ActionWrapper.Timeouted(originalCallTimeoutMillis)) | ||||
|  | ||||
|     protected open suspend fun actualizeAll(): Result<Unit> { | ||||
|         return runCatchingSafely { | ||||
|             kvCache.actualizeAll(originalRepo) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     override suspend fun contains(k: Id): Boolean = actionWrapper.wrap { | ||||
|         originalRepo.contains(k) | ||||
|     }.getOrElse { | ||||
|         kvCache.contains(k) | ||||
|     } | ||||
|  | ||||
|     override suspend fun count(): Long = actionWrapper.wrap { | ||||
|         originalRepo.count() | ||||
|     }.getOrElse { | ||||
|         kvCache.count() | ||||
|     } | ||||
|  | ||||
|     override suspend fun keys( | ||||
|         v: RegisteredObject, | ||||
|         pagination: Pagination, | ||||
|         reversed: Boolean | ||||
|     ): PaginationResult<Id> = actionWrapper.wrap { | ||||
|         originalRepo.keys(v, pagination, reversed) | ||||
|     }.getOrElse { | ||||
|         val results = mutableListOf<Id>() | ||||
|  | ||||
|         val toSkip = pagination.firstIndex | ||||
|         var count = 0 | ||||
|  | ||||
|         doForAllWithNextPaging { | ||||
|             kvCache.keys(pagination, reversed).also { | ||||
|                 it.results.forEach { | ||||
|                     if (kvCache.get(it) ?.contains(v) == true) { | ||||
|                         count++ | ||||
|                         if (count < toSkip || results.size >= pagination.size) { | ||||
|                             return@forEach | ||||
|                         } else { | ||||
|                             results.add(it) | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return@getOrElse results.createPaginationResult( | ||||
|             pagination, | ||||
|             count.toLong() | ||||
|         ) | ||||
|     } | ||||
|  | ||||
|     override suspend fun keys( | ||||
|         pagination: Pagination, | ||||
|         reversed: Boolean | ||||
|     ): PaginationResult<Id> = actionWrapper.wrap { | ||||
|         originalRepo.keys(pagination, reversed) | ||||
|     }.getOrElse { kvCache.keys(pagination, reversed) } | ||||
|  | ||||
|     override suspend fun get( | ||||
|         k: Id, | ||||
|         pagination: Pagination, | ||||
|         reversed: Boolean | ||||
|     ): PaginationResult<RegisteredObject> = actionWrapper.wrap { | ||||
|         originalRepo.get(k, pagination, reversed) | ||||
|     }.getOrNull() ?.also { | ||||
|         it.results.forEach { | ||||
|             kvCache.set(k, ((kvCache.get(k) ?: return@also) + it).distinct()) | ||||
|         } | ||||
|     } ?: kvCache.get(k) ?.run { | ||||
|         paginate(pagination.optionallyReverse(size, reversed)).let { | ||||
|             if (reversed) { | ||||
|                 it.changeResultsUnchecked( | ||||
|                     it.results.reversed() | ||||
|                 ) | ||||
|             } else { | ||||
|                 it | ||||
|             } | ||||
|         } | ||||
|     } ?: emptyPaginationResult() | ||||
|  | ||||
|     override suspend fun count(k: Id): Long = actionWrapper.wrap { | ||||
|         originalRepo.count(k) | ||||
|     }.getOrElse { | ||||
|         kvCache.get(k) ?.size ?.toLong() ?: 0L | ||||
|     } | ||||
|  | ||||
|     override suspend fun contains(k: Id, v: RegisteredObject): Boolean { | ||||
|         return (actionWrapper.wrap { | ||||
|             originalRepo.contains(k, v) | ||||
|         }.getOrNull() ?.also { | ||||
|             kvCache.set(k, ((kvCache.get(k) ?: return@also) + v).distinct()) | ||||
|         }) ?: (kvCache.get(k) ?.contains(v) == true) | ||||
|     } | ||||
|  | ||||
|     override suspend fun invalidate() { | ||||
|         actualizeAll() | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,86 @@ | ||||
| package dev.inmo.micro_utils.repos.cache.fallback.keyvalues | ||||
|  | ||||
| import dev.inmo.micro_utils.coroutines.plus | ||||
| import dev.inmo.micro_utils.coroutines.subscribeSafelyWithoutExceptions | ||||
| import dev.inmo.micro_utils.pagination.FirstPagePagination | ||||
| import dev.inmo.micro_utils.pagination.utils.doForAllWithNextPaging | ||||
| import dev.inmo.micro_utils.repos.WriteKeyValuesRepo | ||||
| import dev.inmo.micro_utils.repos.cache.cache.FullKVCache | ||||
| import dev.inmo.micro_utils.repos.cache.FallbackCacheRepo | ||||
| import dev.inmo.micro_utils.repos.set | ||||
| import dev.inmo.micro_utils.repos.unset | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlinx.coroutines.flow.Flow | ||||
| import kotlinx.coroutines.flow.distinctUntilChanged | ||||
|  | ||||
| open class AutoRecacheWriteKeyValuesRepo<Id, RegisteredObject>( | ||||
|     protected val originalRepo: WriteKeyValuesRepo<Id, RegisteredObject>, | ||||
|     protected val scope: CoroutineScope, | ||||
|     protected val kvCache: FullKVCache<Id, List<RegisteredObject>> = FullKVCache() | ||||
| ) : WriteKeyValuesRepo<Id, RegisteredObject>, FallbackCacheRepo { | ||||
|     override val onValueRemoved: Flow<Pair<Id, RegisteredObject>> | ||||
|         get() = originalRepo.onValueRemoved | ||||
|  | ||||
|     override val onNewValue: Flow<Pair<Id, RegisteredObject>> | ||||
|         get() = originalRepo.onNewValue | ||||
|     override val onDataCleared: Flow<Id> | ||||
|         get() = (originalRepo.onDataCleared + kvCache.onValueRemoved).distinctUntilChanged() | ||||
|  | ||||
|     private val onDataClearedListeningJob = originalRepo.onDataCleared.subscribeSafelyWithoutExceptions(scope) { | ||||
|         kvCache.unset(it) | ||||
|     } | ||||
|  | ||||
|     private val onRemovingUpdatesListeningJob = originalRepo.onValueRemoved.subscribeSafelyWithoutExceptions(scope) { | ||||
|         kvCache.set( | ||||
|             it.first, | ||||
|             (kvCache.get( | ||||
|                 it.first | ||||
|             ) ?: return@subscribeSafelyWithoutExceptions) - it.second | ||||
|         ) | ||||
|     } | ||||
|  | ||||
|     private val onNewAndUpdatedObjectsListeningJob = originalRepo.onNewValue.subscribeSafelyWithoutExceptions(scope) { | ||||
|         kvCache.set( | ||||
|             it.first, | ||||
|             (kvCache.get( | ||||
|                 it.first | ||||
|             ) ?: return@subscribeSafelyWithoutExceptions) + it.second | ||||
|         ) | ||||
|     } | ||||
|  | ||||
|     override suspend fun clearWithValue(v: RegisteredObject) { | ||||
|         originalRepo.clearWithValue(v) | ||||
|         doForAllWithNextPaging(FirstPagePagination(kvCache.count().takeIf { it < Int.MAX_VALUE } ?.toInt() ?: Int.MAX_VALUE)) { | ||||
|             kvCache.keys(it).also { | ||||
|                 it.results.forEach { id -> | ||||
|                     kvCache.get(id) ?.takeIf { it.contains(v) } ?.let { | ||||
|                         kvCache.unset(id) | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     override suspend fun clear(k: Id) { | ||||
|         originalRepo.clear(k) | ||||
|         kvCache.unset(k) | ||||
|     } | ||||
|  | ||||
|     override suspend fun remove(toRemove: Map<Id, List<RegisteredObject>>) { | ||||
|         originalRepo.remove(toRemove) | ||||
|         toRemove.forEach { (k, v) -> | ||||
|             kvCache.set(k, (kvCache.get(k) ?: return@forEach) - v) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     override suspend fun add(toAdd: Map<Id, List<RegisteredObject>>) { | ||||
|         originalRepo.add(toAdd) | ||||
|         toAdd.forEach { (k, v) -> | ||||
|             kvCache.set(k, (kvCache.get(k) ?: return@forEach) + v) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     override suspend fun invalidate() { | ||||
|         kvCache.clear() | ||||
|     } | ||||
| } | ||||
| @@ -3,11 +3,10 @@ package dev.inmo.micro_utils.repos.cache.full | ||||
| import dev.inmo.micro_utils.common.* | ||||
| import dev.inmo.micro_utils.pagination.Pagination | ||||
| import dev.inmo.micro_utils.pagination.PaginationResult | ||||
| import dev.inmo.micro_utils.pagination.utils.doForAllWithNextPaging | ||||
| import dev.inmo.micro_utils.repos.* | ||||
| import dev.inmo.micro_utils.repos.cache.* | ||||
| import dev.inmo.micro_utils.repos.cache.cache.FullKVCache | ||||
| import dev.inmo.micro_utils.repos.cache.cache.KVCache | ||||
| import dev.inmo.micro_utils.repos.cache.util.actualizeAll | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlinx.coroutines.Dispatchers | ||||
|  | ||||
| @@ -32,12 +31,7 @@ open class FullReadCRUDCacheRepo<ObjectType, IdType>( | ||||
|     } | ||||
|  | ||||
|     protected open suspend fun actualizeAll() { | ||||
|         kvCache.clear() | ||||
|         doForAllWithNextPaging { | ||||
|             parentRepo.getByPagination(it).also { | ||||
|                 kvCache.set(it.results.associateBy { idGetter(it) }) | ||||
|             } | ||||
|         } | ||||
|         kvCache.actualizeAll(parentRepo) | ||||
|     } | ||||
|  | ||||
|     override suspend fun getByPagination(pagination: Pagination): PaginationResult<ObjectType> = doOrTakeAndActualize( | ||||
| @@ -69,6 +63,10 @@ open class FullReadCRUDCacheRepo<ObjectType, IdType>( | ||||
|         { getById(id) }, | ||||
|         { it ?.let { set(idGetter(it), it) } } | ||||
|     ) | ||||
|  | ||||
|     override suspend fun invalidate() { | ||||
|         actualizeAll() | ||||
|     } | ||||
| } | ||||
|  | ||||
| fun <ObjectType, IdType> ReadCRUDRepo<ObjectType, IdType>.cached( | ||||
| @@ -92,7 +90,11 @@ open class FullCRUDCacheRepo<ObjectType, IdType, InputValueType>( | ||||
|         scope, | ||||
|         idGetter | ||||
|     ), | ||||
|     CRUDRepo<ObjectType, IdType, InputValueType> | ||||
|     CRUDRepo<ObjectType, IdType, InputValueType> { | ||||
|     override suspend fun invalidate() { | ||||
|         actualizeAll() | ||||
|     } | ||||
| } | ||||
|  | ||||
| fun <ObjectType, IdType, InputType> CRUDRepo<ObjectType, IdType, InputType>.cached( | ||||
|     kvCache: FullKVCache<IdType, ObjectType>, | ||||
|   | ||||
| @@ -5,9 +5,11 @@ import dev.inmo.micro_utils.pagination.Pagination | ||||
| import dev.inmo.micro_utils.pagination.PaginationResult | ||||
| import dev.inmo.micro_utils.repos.* | ||||
| import dev.inmo.micro_utils.repos.cache.cache.FullKVCache | ||||
| import dev.inmo.micro_utils.repos.cache.util.actualizeAll | ||||
| import dev.inmo.micro_utils.repos.pagination.getAll | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlinx.coroutines.Dispatchers | ||||
| import kotlinx.coroutines.Job | ||||
| import kotlinx.coroutines.flow.* | ||||
|  | ||||
| open class FullReadKeyValueCacheRepo<Key,Value>( | ||||
| @@ -68,6 +70,10 @@ open class FullReadKeyValueCacheRepo<Key,Value>( | ||||
|         { parentRepo.keys(v, pagination, reversed) }, | ||||
|         { if (it.results.isNotEmpty()) actualizeAll() } | ||||
|     ) | ||||
|  | ||||
|     override suspend fun invalidate() { | ||||
|         actualizeAll() | ||||
|     } | ||||
| } | ||||
|  | ||||
| fun <Key, Value> ReadKeyValueRepo<Key, Value>.cached( | ||||
| @@ -75,12 +81,16 @@ fun <Key, Value> ReadKeyValueRepo<Key, Value>.cached( | ||||
| ) = FullReadKeyValueCacheRepo(this, kvCache) | ||||
|  | ||||
| open class FullWriteKeyValueCacheRepo<Key,Value>( | ||||
|     protected open val parentRepo: WriteKeyValueRepo<Key, Value>, | ||||
|     parentRepo: WriteKeyValueRepo<Key, Value>, | ||||
|     protected open val kvCache: FullKVCache<Key, Value>, | ||||
|     scope: CoroutineScope = CoroutineScope(Dispatchers.Default) | ||||
| ) : WriteKeyValueRepo<Key, Value> by parentRepo, FullCacheRepo { | ||||
|     protected val onNewJob = parentRepo.onNewValue.onEach { kvCache.set(it.first, it.second) }.launchIn(scope) | ||||
|     protected val onRemoveJob = parentRepo.onValueRemoved.onEach { kvCache.unset(it) }.launchIn(scope) | ||||
|  | ||||
|     override suspend fun invalidate() { | ||||
|         kvCache.clear() | ||||
|     } | ||||
| } | ||||
|  | ||||
| fun <Key, Value> WriteKeyValueRepo<Key, Value>.caching( | ||||
| @@ -89,13 +99,17 @@ fun <Key, Value> WriteKeyValueRepo<Key, Value>.caching( | ||||
| ) = FullWriteKeyValueCacheRepo(this, kvCache, scope) | ||||
|  | ||||
| open class FullKeyValueCacheRepo<Key,Value>( | ||||
|     parentRepo: KeyValueRepo<Key, Value>, | ||||
|     protected open val parentRepo: KeyValueRepo<Key, Value>, | ||||
|     kvCache: FullKVCache<Key, Value>, | ||||
|     scope: CoroutineScope = CoroutineScope(Dispatchers.Default) | ||||
| ) : FullWriteKeyValueCacheRepo<Key,Value>(parentRepo, kvCache, scope), | ||||
|     KeyValueRepo<Key,Value>, | ||||
|     ReadKeyValueRepo<Key, Value> by FullReadKeyValueCacheRepo(parentRepo, kvCache) { | ||||
|     override suspend fun unsetWithValues(toUnset: List<Value>) = parentRepo.unsetWithValues(toUnset) | ||||
|  | ||||
|     override suspend fun invalidate() { | ||||
|         kvCache.actualizeAll(parentRepo) | ||||
|     } | ||||
| } | ||||
|  | ||||
| fun <Key, Value> KeyValueRepo<Key, Value>.cached( | ||||
|   | ||||
| @@ -5,6 +5,7 @@ import dev.inmo.micro_utils.pagination.* | ||||
| import dev.inmo.micro_utils.pagination.utils.* | ||||
| import dev.inmo.micro_utils.repos.* | ||||
| import dev.inmo.micro_utils.repos.cache.cache.FullKVCache | ||||
| import dev.inmo.micro_utils.repos.cache.util.actualizeAll | ||||
| import kotlinx.coroutines.CoroutineScope | ||||
| import kotlinx.coroutines.Dispatchers | ||||
| import kotlinx.coroutines.flow.* | ||||
| @@ -33,8 +34,7 @@ open class FullReadKeyValuesCacheRepo<Key,Value>( | ||||
|     } | ||||
|  | ||||
|     protected open suspend fun actualizeAll() { | ||||
|         doAllWithCurrentPaging { kvCache.keys(it).also { kvCache.unset(it.results) } } | ||||
|         kvCache.set(parentRepo.getAll()) | ||||
|         kvCache.actualizeAll(parentRepo) | ||||
|     } | ||||
|  | ||||
|     override suspend fun get(k: Key, pagination: Pagination, reversed: Boolean): PaginationResult<Value> { | ||||
| @@ -102,6 +102,9 @@ open class FullReadKeyValuesCacheRepo<Key,Value>( | ||||
|         { if (it.results.isNotEmpty()) actualizeAll() } | ||||
|     ) | ||||
|  | ||||
|     override suspend fun invalidate() { | ||||
|         actualizeAll() | ||||
|     } | ||||
| } | ||||
|  | ||||
| fun <Key, Value> ReadKeyValuesRepo<Key, Value>.cached( | ||||
| @@ -109,7 +112,7 @@ fun <Key, Value> ReadKeyValuesRepo<Key, Value>.cached( | ||||
| ) = FullReadKeyValuesCacheRepo(this, kvCache) | ||||
|  | ||||
| open class FullWriteKeyValuesCacheRepo<Key,Value>( | ||||
|     protected open val parentRepo: WriteKeyValuesRepo<Key, Value>, | ||||
|     parentRepo: WriteKeyValuesRepo<Key, Value>, | ||||
|     protected open val kvCache: FullKVCache<Key, List<Value>>, | ||||
|     scope: CoroutineScope = CoroutineScope(Dispatchers.Default) | ||||
| ) : WriteKeyValuesRepo<Key, Value> by parentRepo, FullCacheRepo { | ||||
| @@ -125,6 +128,10 @@ open class FullWriteKeyValuesCacheRepo<Key,Value>( | ||||
|             kvCache.get(it.first) ?.minus(it.second) ?: return@onEach | ||||
|         ) | ||||
|     }.launchIn(scope) | ||||
|  | ||||
|     override suspend fun invalidate() { | ||||
|         kvCache.clear() | ||||
|     } | ||||
| } | ||||
|  | ||||
| fun <Key, Value> WriteKeyValuesRepo<Key, Value>.caching( | ||||
| @@ -133,7 +140,7 @@ fun <Key, Value> WriteKeyValuesRepo<Key, Value>.caching( | ||||
| ) = FullWriteKeyValuesCacheRepo(this, kvCache, scope) | ||||
|  | ||||
| open class FullKeyValuesCacheRepo<Key,Value>( | ||||
|     parentRepo: KeyValuesRepo<Key, Value>, | ||||
|     protected open val parentRepo: KeyValuesRepo<Key, Value>, | ||||
|     kvCache: FullKVCache<Key, List<Value>>, | ||||
|     scope: CoroutineScope = CoroutineScope(Dispatchers.Default) | ||||
| ) : FullWriteKeyValuesCacheRepo<Key, Value>(parentRepo, kvCache, scope), | ||||
| @@ -146,6 +153,10 @@ open class FullKeyValuesCacheRepo<Key,Value>( | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     override suspend fun invalidate() { | ||||
|         kvCache.actualizeAll(parentRepo) | ||||
|     } | ||||
| } | ||||
|  | ||||
| fun <Key, Value> KeyValuesRepo<Key, Value>.caching( | ||||
|   | ||||
							
								
								
									
										53
									
								
								repos/cache/src/commonMain/kotlin/dev/inmo/micro_utils/repos/cache/util/ActualizeAll.kt
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										53
									
								
								repos/cache/src/commonMain/kotlin/dev/inmo/micro_utils/repos/cache/util/ActualizeAll.kt
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,53 @@ | ||||
| package dev.inmo.micro_utils.repos.cache.util | ||||
|  | ||||
| import dev.inmo.micro_utils.pagination.FirstPagePagination | ||||
| import dev.inmo.micro_utils.pagination.utils.doForAllWithNextPaging | ||||
| import dev.inmo.micro_utils.pagination.utils.getAllByWithNextPaging | ||||
| import dev.inmo.micro_utils.repos.ReadCRUDRepo | ||||
| import dev.inmo.micro_utils.repos.ReadKeyValueRepo | ||||
| import dev.inmo.micro_utils.repos.ReadKeyValuesRepo | ||||
| import dev.inmo.micro_utils.repos.cache.cache.KVCache | ||||
| import dev.inmo.micro_utils.repos.pagination.getAll | ||||
| import dev.inmo.micro_utils.repos.set | ||||
|  | ||||
| suspend inline fun <K, V> KVCache<K, V>.actualizeAll( | ||||
|     clear: Boolean = true, | ||||
|     getAll: () -> Map<K, V> | ||||
| ) { | ||||
|     set( | ||||
|         getAll().also { | ||||
|             if (clear) { | ||||
|                 clear() | ||||
|             } | ||||
|         } | ||||
|     ) | ||||
| } | ||||
|  | ||||
| suspend inline fun <K, V> KVCache<K, V>.actualizeAll( | ||||
|     repo: ReadKeyValueRepo<K, V>, | ||||
|     clear: Boolean = true, | ||||
| ) { | ||||
|     actualizeAll(clear) { | ||||
|         repo.getAll { keys(it) }.toMap() | ||||
|     } | ||||
| } | ||||
|  | ||||
| suspend inline fun <K, V> KVCache<K, List<V>>.actualizeAll( | ||||
|     repo: ReadKeyValuesRepo<K, V>, | ||||
|     clear: Boolean = true, | ||||
| ) { | ||||
|     actualizeAll(clear) { | ||||
|         repo.getAll { keys(it) }.toMap() | ||||
|     } | ||||
| } | ||||
|  | ||||
| suspend inline fun <K, V> KVCache<K, V>.actualizeAll( | ||||
|     repo: ReadCRUDRepo<V, K>, | ||||
|     clear: Boolean = true, | ||||
| ) { | ||||
|     actualizeAll(clear) { | ||||
|         repo.getAllByWithNextPaging { | ||||
|             getIdsByPagination(it) | ||||
|         }.mapNotNull { it to (repo.getById(it) ?: return@mapNotNull null) }.toMap() | ||||
|     } | ||||
| } | ||||
| @@ -2,6 +2,8 @@ package dev.inmo.micro_utils.repos | ||||
|  | ||||
| import dev.inmo.micro_utils.pagination.* | ||||
| import dev.inmo.micro_utils.pagination.utils.doAllWithCurrentPaging | ||||
| import dev.inmo.micro_utils.pagination.utils.getAllWithNextPaging | ||||
| import dev.inmo.micro_utils.pagination.utils.paginate | ||||
| import kotlinx.coroutines.flow.Flow | ||||
|  | ||||
| /** | ||||
| @@ -17,24 +19,32 @@ interface ReadKeyValueRepo<Key, Value> : Repo { | ||||
|     suspend fun get(k: Key): Value? | ||||
|  | ||||
|     /** | ||||
|      * This method should use sorted by [Key]s search and take the [PaginationResult]. By default, it should use | ||||
|      * This method should use sorted by [Key]s search and return the [PaginationResult]. By default, it should use | ||||
|      * ascending sort for [Key]s | ||||
|      */ | ||||
|     suspend fun values(pagination: Pagination, reversed: Boolean = false): PaginationResult<Value> | ||||
|  | ||||
|     /** | ||||
|      * This method should use sorted by [Key]s search and take the [PaginationResult]. By default, it should use | ||||
|      * This method should use sorted by [Key]s search and return the [PaginationResult]. By default, it should use | ||||
|      * ascending sort for [Key]s | ||||
|      */ | ||||
|     suspend fun keys(pagination: Pagination, reversed: Boolean = false): PaginationResult<Key> | ||||
|  | ||||
|     /** | ||||
|      * This method should use sorted by [Key]s search and take the [PaginationResult]. By default, it should use | ||||
|      * ascending sort for [Key]s | ||||
|      * This method should use sorted by [Key]s search and return the [PaginationResult]. By default, it should use | ||||
|      * ascending sort for [Key]s. | ||||
|      * | ||||
|      * **DEFAULT REALIZATION IS NOT OPTIMAL AND HAS BEEN ADDED TO COVER CASES OF DIFFERENT COMMON MAPPINGS AND TRANSFORMATIONS** | ||||
|      * | ||||
|      * @param v This value should be used to exclude from search the items with different [Value]s | ||||
|      */ | ||||
|     suspend fun keys(v: Value, pagination: Pagination, reversed: Boolean = false): PaginationResult<Key> | ||||
|     suspend fun keys(v: Value, pagination: Pagination, reversed: Boolean = false): PaginationResult<Key> { | ||||
|         return getAllWithNextPaging { | ||||
|             keys(it) | ||||
|         }.filter { | ||||
|             get(it) == v | ||||
|         }.paginate(pagination, reversed) | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @return true if [key] is presented in current collection or false otherwise | ||||
| @@ -93,6 +103,10 @@ suspend inline fun <Key, Value> WriteKeyValueRepo<Key, Value>.set( | ||||
|     vararg toSet: Pair<Key, Value> | ||||
| ) = set(toSet.toMap()) | ||||
|  | ||||
| suspend inline fun <Key, Value> WriteKeyValueRepo<Key, Value>.set( | ||||
|     toSet: List<Pair<Key, Value>> | ||||
| ) = set(toSet.toMap()) | ||||
|  | ||||
| suspend inline fun <Key, Value> WriteKeyValueRepo<Key, Value>.set( | ||||
|     k: Key, v: Value | ||||
| ) = set(k to v) | ||||
| @@ -125,7 +139,11 @@ interface KeyValueRepo<Key, Value> : ReadKeyValueRepo<Key, Value>, WriteKeyValue | ||||
|      * By default, will remove all the data of current repo using [doAllWithCurrentPaging], [keys] and [unset] | ||||
|      */ | ||||
|     suspend fun clear() { | ||||
|         doAllWithCurrentPaging { keys(it).also { unset(it.results) } } | ||||
|         var count: Int | ||||
|         do { | ||||
|             count = count().takeIf { it < Int.MAX_VALUE } ?.toInt() ?: Int.MAX_VALUE | ||||
|             keys(FirstPagePagination(count)).also { unset(it.results) } | ||||
|         } while(count > 0) | ||||
|     } | ||||
| } | ||||
| typealias StandardKeyValueRepo<Key,Value> = KeyValueRepo<Key, Value> | ||||
|   | ||||
| @@ -0,0 +1,10 @@ | ||||
| package dev.inmo.micro_utils.repos.transforms.crud | ||||
|  | ||||
| import dev.inmo.micro_utils.repos.KeyValueRepo | ||||
| import dev.inmo.micro_utils.repos.KeyValuesRepo | ||||
| import dev.inmo.micro_utils.repos.ReadKeyValueRepo | ||||
| import dev.inmo.micro_utils.repos.ReadKeyValuesRepo | ||||
| import kotlin.js.JsName | ||||
| import kotlin.jvm.JvmName | ||||
|  | ||||
| fun <K, V> ReadKeyValueRepo<K, V>.asReadCRUDRepo() = ReadCRUDFromKeyValueRepo(this) | ||||
| @@ -0,0 +1,20 @@ | ||||
| package dev.inmo.micro_utils.repos.transforms.crud | ||||
|  | ||||
| import dev.inmo.micro_utils.pagination.Pagination | ||||
| import dev.inmo.micro_utils.pagination.PaginationResult | ||||
| import dev.inmo.micro_utils.repos.ReadCRUDRepo | ||||
| import dev.inmo.micro_utils.repos.ReadKeyValueRepo | ||||
|  | ||||
| open class ReadCRUDFromKeyValueRepo<RegisteredType, IdType>( | ||||
|     protected open val original: ReadKeyValueRepo<IdType, RegisteredType> | ||||
| ) : ReadCRUDRepo<RegisteredType, IdType> { | ||||
|     override suspend fun contains(id: IdType): Boolean = original.contains(id) | ||||
|  | ||||
|     override suspend fun count(): Long = original.count() | ||||
|  | ||||
|     override suspend fun getByPagination(pagination: Pagination): PaginationResult<RegisteredType> = original.values(pagination) | ||||
|  | ||||
|     override suspend fun getIdsByPagination(pagination: Pagination): PaginationResult<IdType> = original.keys(pagination) | ||||
|  | ||||
|     override suspend fun getById(id: IdType): RegisteredType? = original.get(id) | ||||
| } | ||||
| @@ -0,0 +1,15 @@ | ||||
| package dev.inmo.micro_utils.repos.transforms.kv | ||||
|  | ||||
| import dev.inmo.micro_utils.repos.KeyValueRepo | ||||
| import dev.inmo.micro_utils.repos.KeyValuesRepo | ||||
| import dev.inmo.micro_utils.repos.ReadCRUDRepo | ||||
| import dev.inmo.micro_utils.repos.ReadKeyValueRepo | ||||
| import dev.inmo.micro_utils.repos.ReadKeyValuesRepo | ||||
| import kotlin.js.JsName | ||||
| import kotlin.jvm.JvmName | ||||
|  | ||||
| fun <K, V> ReadKeyValuesRepo<K, V>.asReadKeyValueRepo() = ReadKeyValueFromKeyValuesRepo(this) | ||||
|  | ||||
| fun <K, V> KeyValuesRepo<K, V>.asKeyValueRepo() = KeyValueFromKeyValuesRepo(this) | ||||
|  | ||||
| fun <K, V> ReadCRUDRepo<K, V>.asReadKeyValueRepo() = ReadKeyValueFromCRUDRepo(this) | ||||
| @@ -0,0 +1,41 @@ | ||||
| package dev.inmo.micro_utils.repos.transforms.kv | ||||
|  | ||||
| import dev.inmo.micro_utils.pagination.FirstPagePagination | ||||
| import dev.inmo.micro_utils.pagination.Pagination | ||||
| import dev.inmo.micro_utils.pagination.PaginationResult | ||||
| import dev.inmo.micro_utils.pagination.changeResults | ||||
| import dev.inmo.micro_utils.pagination.changeResultsUnchecked | ||||
| import dev.inmo.micro_utils.pagination.utils.doForAllWithNextPaging | ||||
| import dev.inmo.micro_utils.pagination.utils.optionallyReverse | ||||
| import dev.inmo.micro_utils.pagination.utils.paginate | ||||
| import dev.inmo.micro_utils.repos.KeyValueRepo | ||||
| import dev.inmo.micro_utils.repos.KeyValuesRepo | ||||
| import dev.inmo.micro_utils.repos.ReadKeyValueRepo | ||||
| import dev.inmo.micro_utils.repos.ReadKeyValuesRepo | ||||
| import dev.inmo.micro_utils.repos.transforms.kvs.ReadKeyValuesFromKeyValueRepo | ||||
| import kotlinx.coroutines.flow.Flow | ||||
| import kotlinx.coroutines.flow.map | ||||
| import kotlinx.coroutines.flow.mapNotNull | ||||
| import kotlinx.coroutines.flow.merge | ||||
|  | ||||
| open class KeyValueFromKeyValuesRepo<Key, Value>( | ||||
|     private val original: KeyValuesRepo<Key, Value> | ||||
| ) : KeyValueRepo<Key, List<Value>>, ReadKeyValueFromKeyValuesRepo<Key, Value>(original) { | ||||
|     override val onNewValue: Flow<Pair<Key, List<Value>>> = merge( | ||||
|         original.onNewValue, | ||||
|         original.onValueRemoved | ||||
|     ).mapNotNull { | ||||
|         it.first to (get(it.first) ?: return@mapNotNull null) | ||||
|     } | ||||
|     override val onValueRemoved: Flow<Key> = original.onDataCleared | ||||
|  | ||||
|     override suspend fun unset(toUnset: List<Key>) { | ||||
|         toUnset.forEach { | ||||
|             original.clear(it) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     override suspend fun set(toSet: Map<Key, List<Value>>) { | ||||
|         original.set(toSet) | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,46 @@ | ||||
| package dev.inmo.micro_utils.repos.transforms.kv | ||||
|  | ||||
| import dev.inmo.micro_utils.pagination.FirstPagePagination | ||||
| import dev.inmo.micro_utils.pagination.Pagination | ||||
| import dev.inmo.micro_utils.pagination.PaginationResult | ||||
| import dev.inmo.micro_utils.pagination.changeResults | ||||
| import dev.inmo.micro_utils.pagination.changeResultsUnchecked | ||||
| import dev.inmo.micro_utils.pagination.utils.doForAllWithNextPaging | ||||
| import dev.inmo.micro_utils.pagination.utils.optionallyReverse | ||||
| import dev.inmo.micro_utils.pagination.utils.paginate | ||||
| import dev.inmo.micro_utils.repos.ReadCRUDRepo | ||||
| import dev.inmo.micro_utils.repos.ReadKeyValueRepo | ||||
| import dev.inmo.micro_utils.repos.ReadKeyValuesRepo | ||||
| import dev.inmo.micro_utils.repos.transforms.kvs.ReadKeyValuesFromKeyValueRepo | ||||
| import kotlin.jvm.JvmInline | ||||
|  | ||||
| @JvmInline | ||||
| value class ReadKeyValueFromCRUDRepo<Key, Value>( | ||||
|     private val original: ReadCRUDRepo<Value, Key> | ||||
| ) : ReadKeyValueRepo<Key, Value> { | ||||
|     override suspend fun get(k: Key): Value? = original.getById(k) | ||||
|  | ||||
|     override suspend fun values(pagination: Pagination, reversed: Boolean): PaginationResult<Value> = original.getByPagination( | ||||
|         pagination.optionallyReverse(count(), reversed) | ||||
|     ).let { | ||||
|         if (reversed) { | ||||
|             it.changeResultsUnchecked(it.results.reversed()) | ||||
|         } else { | ||||
|             it | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     override suspend fun keys(pagination: Pagination, reversed: Boolean): PaginationResult<Key> = original.getIdsByPagination( | ||||
|         pagination.optionallyReverse(count(), reversed) | ||||
|     ).let { | ||||
|         if (reversed) { | ||||
|             it.changeResultsUnchecked(it.results.reversed()) | ||||
|         } else { | ||||
|             it | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     override suspend fun count(): Long = original.count() | ||||
|  | ||||
|     override suspend fun contains(key: Key): Boolean = original.contains(key) | ||||
| } | ||||
| @@ -0,0 +1,67 @@ | ||||
| package dev.inmo.micro_utils.repos.transforms.kv | ||||
|  | ||||
| import dev.inmo.micro_utils.pagination.FirstPagePagination | ||||
| import dev.inmo.micro_utils.pagination.Pagination | ||||
| import dev.inmo.micro_utils.pagination.PaginationResult | ||||
| import dev.inmo.micro_utils.pagination.changeResults | ||||
| import dev.inmo.micro_utils.pagination.changeResultsUnchecked | ||||
| import dev.inmo.micro_utils.pagination.utils.doForAllWithNextPaging | ||||
| import dev.inmo.micro_utils.pagination.utils.optionallyReverse | ||||
| import dev.inmo.micro_utils.pagination.utils.paginate | ||||
| import dev.inmo.micro_utils.repos.ReadKeyValueRepo | ||||
| import dev.inmo.micro_utils.repos.ReadKeyValuesRepo | ||||
| import dev.inmo.micro_utils.repos.transforms.kvs.ReadKeyValuesFromKeyValueRepo | ||||
|  | ||||
| open class ReadKeyValueFromKeyValuesRepo<Key, Value>( | ||||
|     private val original: ReadKeyValuesRepo<Key, Value> | ||||
| ) : ReadKeyValueRepo<Key, List<Value>> { | ||||
|     override suspend fun get(k: Key): List<Value>? = original.getAll(k) | ||||
|  | ||||
|     override suspend fun values( | ||||
|         pagination: Pagination, | ||||
|         reversed: Boolean | ||||
|     ): PaginationResult<List<Value>> { | ||||
|         val keys = keys(pagination, reversed) | ||||
|         return keys.changeResults( | ||||
|             keys.results.mapNotNull { | ||||
|                 get(it) | ||||
|             } | ||||
|         ) | ||||
|     } | ||||
|  | ||||
|     override suspend fun keys(pagination: Pagination, reversed: Boolean): PaginationResult<Key> { | ||||
|         return original.keys(pagination, reversed) | ||||
|     } | ||||
|  | ||||
|     override suspend fun count(): Long { | ||||
|         return original.count() | ||||
|     } | ||||
|  | ||||
|     override suspend fun contains(key: Key): Boolean { | ||||
|         return original.contains(key) | ||||
|     } | ||||
|  | ||||
|     override suspend fun keys(v: List<Value>, pagination: Pagination, reversed: Boolean): PaginationResult<Key> { | ||||
|         val keys = mutableSetOf<Key>() | ||||
|  | ||||
|         doForAllWithNextPaging(FirstPagePagination(count().toInt())) { | ||||
|             original.keys(it).also { | ||||
|                 it.results.forEach { | ||||
|                     val values = get(it) ?: return@forEach | ||||
|                     if (values.containsAll(v) && v.containsAll(values)) { | ||||
|                         keys.add(it) | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         val paginated = keys.paginate( | ||||
|             pagination.optionallyReverse(keys.count(), reversed) | ||||
|         ) | ||||
|         return if (reversed) { | ||||
|             paginated.changeResultsUnchecked(paginated.results.reversed()) | ||||
|         } else { | ||||
|             paginated | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,21 @@ | ||||
| package dev.inmo.micro_utils.repos.transforms.kvs | ||||
|  | ||||
| import dev.inmo.micro_utils.repos.KeyValueRepo | ||||
| import dev.inmo.micro_utils.repos.ReadKeyValueRepo | ||||
| import kotlin.js.JsName | ||||
| import kotlin.jvm.JvmName | ||||
|  | ||||
|  | ||||
| fun <K, V, VI : Iterable<V>> ReadKeyValueRepo<K, VI>.asReadKeyValuesRepo() = ReadKeyValuesFromKeyValueRepo(this) | ||||
|  | ||||
| fun <K, V, VI : Iterable<V>> KeyValueRepo<K, VI>.asKeyValuesRepo( | ||||
|     listToValuesIterable: suspend (List<V>) -> VI | ||||
| ): KeyValuesFromKeyValueRepo<K, V, VI> = KeyValuesFromKeyValueRepo(this, listToValuesIterable) | ||||
|  | ||||
| @JvmName("asListKeyValuesRepo") | ||||
| @JsName("asListKeyValuesRepo") | ||||
| fun <K, V> KeyValueRepo<K, List<V>>.asKeyValuesRepo(): KeyValuesFromKeyValueRepo<K, V, List<V>> = asKeyValuesRepo { it } | ||||
|  | ||||
| @JvmName("asSetKeyValuesRepo") | ||||
| @JsName("asSetKeyValuesRepo") | ||||
| fun <K, V> KeyValueRepo<K, Set<V>>.asKeyValuesRepo(): KeyValuesFromKeyValueRepo<K, V, Set<V>> = asKeyValuesRepo { it.toSet() } | ||||
| @@ -0,0 +1,77 @@ | ||||
| package dev.inmo.micro_utils.repos.transforms.kvs | ||||
|  | ||||
| import dev.inmo.micro_utils.pagination.FirstPagePagination | ||||
| import dev.inmo.micro_utils.pagination.utils.doForAllWithNextPaging | ||||
| import dev.inmo.micro_utils.repos.KeyValueRepo | ||||
| import dev.inmo.micro_utils.repos.KeyValuesRepo | ||||
| import dev.inmo.micro_utils.repos.unset | ||||
| import kotlinx.coroutines.flow.Flow | ||||
| import kotlinx.coroutines.flow.MutableSharedFlow | ||||
| import kotlinx.coroutines.flow.asSharedFlow | ||||
| import kotlin.js.JsName | ||||
| import kotlin.jvm.JvmName | ||||
|  | ||||
| open class KeyValuesFromKeyValueRepo<Key, Value, ValuesIterable : Iterable<Value>>( | ||||
|     private val original: KeyValueRepo<Key, ValuesIterable>, | ||||
|     private val listToValuesIterable: suspend (List<Value>) -> ValuesIterable | ||||
| ) : KeyValuesRepo<Key, Value>, ReadKeyValuesFromKeyValueRepo<Key, Value, ValuesIterable>(original) { | ||||
|     private val _onNewValue = MutableSharedFlow<Pair<Key, Value>>() | ||||
|     private val _onValueRemoved = MutableSharedFlow<Pair<Key, Value>>() | ||||
|     override val onNewValue: Flow<Pair<Key, Value>> = _onNewValue.asSharedFlow() | ||||
|     override val onValueRemoved: Flow<Pair<Key, Value>> = _onValueRemoved.asSharedFlow() | ||||
|     override val onDataCleared: Flow<Key> = original.onValueRemoved | ||||
|  | ||||
|     override suspend fun clearWithValue(v: Value) { | ||||
|         val keys = mutableSetOf<Key>() | ||||
|  | ||||
|         doForAllWithNextPaging(FirstPagePagination(count().toInt())) { | ||||
|             original.keys(it).also { | ||||
|                 it.results.forEach { | ||||
|                     if (contains(it, v)) { | ||||
|                         keys.add(it) | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         original.unset(keys.toList()) | ||||
|     } | ||||
|  | ||||
|     override suspend fun clear(k: Key) { | ||||
|         original.unset(k) | ||||
|     } | ||||
|  | ||||
|     override suspend fun remove(toRemove: Map<Key, List<Value>>) { | ||||
|         original.set( | ||||
|             toRemove.mapNotNull { (k, removing) -> | ||||
|                 val exists = original.get(k) ?: return@mapNotNull null | ||||
|                 k to listToValuesIterable(exists - removing).also { | ||||
|                     if (it.firstOrNull() == null) { | ||||
|                         original.unset(k) | ||||
|                         return@mapNotNull null | ||||
|                     } | ||||
|                 } | ||||
|             }.toMap() | ||||
|         ) | ||||
|         toRemove.forEach { (k, v) -> | ||||
|             v.forEach { | ||||
|                 _onValueRemoved.emit(k to it) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     override suspend fun add(toAdd: Map<Key, List<Value>>) { | ||||
|         original.set( | ||||
|             toAdd.mapNotNull { (k, adding) -> | ||||
|                 val exists = original.get(k) ?: emptyList() | ||||
|                 k to listToValuesIterable(exists + adding) | ||||
|             }.toMap() | ||||
|         ) | ||||
|         toAdd.forEach { (k, v) -> | ||||
|             v.forEach { | ||||
|                 _onNewValue.emit(k to it) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -0,0 +1,72 @@ | ||||
| package dev.inmo.micro_utils.repos.transforms.kvs | ||||
|  | ||||
| import dev.inmo.micro_utils.pagination.FirstPagePagination | ||||
| import dev.inmo.micro_utils.pagination.Pagination | ||||
| import dev.inmo.micro_utils.pagination.PaginationResult | ||||
| import dev.inmo.micro_utils.pagination.changeResultsUnchecked | ||||
| import dev.inmo.micro_utils.pagination.emptyPaginationResult | ||||
| import dev.inmo.micro_utils.pagination.utils.doForAllWithNextPaging | ||||
| import dev.inmo.micro_utils.pagination.utils.optionallyReverse | ||||
| import dev.inmo.micro_utils.pagination.utils.paginate | ||||
| import dev.inmo.micro_utils.repos.ReadKeyValueRepo | ||||
| import dev.inmo.micro_utils.repos.ReadKeyValuesRepo | ||||
|  | ||||
| open class ReadKeyValuesFromKeyValueRepo<Key, Value, ValuesIterable : Iterable<Value>>( | ||||
|     private val original: ReadKeyValueRepo<Key, ValuesIterable> | ||||
| ) : ReadKeyValuesRepo<Key, Value> { | ||||
|     override suspend fun get( | ||||
|         k: Key, | ||||
|         pagination: Pagination, | ||||
|         reversed: Boolean | ||||
|     ): PaginationResult<Value> { | ||||
|         val iterable = original.get(k) ?: return emptyPaginationResult(pagination) | ||||
|         val paginated = iterable.paginate( | ||||
|             pagination.optionallyReverse(iterable.count(), reversed) | ||||
|         ) | ||||
|         return if (reversed) { | ||||
|             paginated.changeResultsUnchecked(paginated.results.reversed()) | ||||
|         } else { | ||||
|             paginated | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     override suspend fun keys( | ||||
|         pagination: Pagination, | ||||
|         reversed: Boolean | ||||
|     ): PaginationResult<Key> = original.keys(pagination, reversed) | ||||
|  | ||||
|     override suspend fun count(): Long = original.count() | ||||
|  | ||||
|     override suspend fun count(k: Key): Long = original.get(k) ?.count() ?.toLong() ?: 0L | ||||
|  | ||||
|     override suspend fun contains(k: Key, v: Value): Boolean = original.get(k) ?.contains(v) == true | ||||
|  | ||||
|     override suspend fun contains(k: Key): Boolean = original.contains(k) | ||||
|  | ||||
|     override suspend fun keys( | ||||
|         v: Value, | ||||
|         pagination: Pagination, | ||||
|         reversed: Boolean | ||||
|     ): PaginationResult<Key> { | ||||
|         val keys = mutableSetOf<Key>() | ||||
|  | ||||
|         doForAllWithNextPaging(FirstPagePagination(count().toInt())) { | ||||
|             original.keys(it).also { | ||||
|                 it.results.forEach { | ||||
|                     if (contains(it, v)) { | ||||
|                         keys.add(it) | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         val paginated = keys.paginate( | ||||
|             pagination.optionallyReverse(keys.count(), reversed) | ||||
|         ) | ||||
|         return if (reversed) { | ||||
|             paginated.changeResultsUnchecked(paginated.results.reversed()) | ||||
|         } else { | ||||
|             paginated | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -31,11 +31,19 @@ class FileReadKeyValueRepo( | ||||
|     override suspend fun values(pagination: Pagination, reversed: Boolean): PaginationResult<File> { | ||||
|         val count = count() | ||||
|         val resultPagination = if (reversed) pagination.reverse(count) else pagination | ||||
|         val filesPaths = folder.list() ?.copyOfRange(resultPagination.firstIndex, resultPagination.lastIndexExclusive) ?: return emptyPaginationResult() | ||||
|         val filesList = folder.list() | ||||
|         val files: Array<String> = if (resultPagination.firstIndex < count) { | ||||
|             val filesPaths = filesList.copyOfRange(resultPagination.firstIndex, resultPagination.lastIndexExclusive.coerceAtMost(filesList.size)) | ||||
|  | ||||
|             if (reversed) { | ||||
|             filesPaths.reverse() | ||||
|                 filesPaths.reversedArray() | ||||
|             } else { | ||||
|                 filesPaths | ||||
|             } | ||||
|         return filesPaths.map { File(folder, it) }.createPaginationResult( | ||||
|         } else { | ||||
|             emptyArray<String>() | ||||
|         } | ||||
|         return files.map { File(folder, it) }.createPaginationResult( | ||||
|             resultPagination, | ||||
|             count | ||||
|         ) | ||||
| @@ -44,11 +52,21 @@ class FileReadKeyValueRepo( | ||||
|     override suspend fun keys(pagination: Pagination, reversed: Boolean): PaginationResult<String> { | ||||
|         val count = count() | ||||
|         val resultPagination = if (reversed) pagination.reverse(count) else pagination | ||||
|         val filesPaths = folder.list() ?.copyOfRange(resultPagination.firstIndex, resultPagination.lastIndexExclusive) ?: return emptyPaginationResult() | ||||
|         val filesList = folder.list() | ||||
|  | ||||
|         val files: Array<String> = if (resultPagination.firstIndex < count) { | ||||
|             val filesPaths = filesList.copyOfRange(resultPagination.firstIndex, resultPagination.lastIndexExclusive.coerceAtMost(filesList.size)) | ||||
|  | ||||
|             if (reversed) { | ||||
|             filesPaths.reverse() | ||||
|                 filesPaths.reversedArray() | ||||
|             } else { | ||||
|                 filesPaths | ||||
|             } | ||||
|         return filesPaths.toList().createPaginationResult( | ||||
|         } else { | ||||
|             emptyArray<String>() | ||||
|         } | ||||
|  | ||||
|         return files.toList().createPaginationResult( | ||||
|             resultPagination, | ||||
|             count | ||||
|         ) | ||||
|   | ||||
| @@ -0,0 +1,23 @@ | ||||
| package dev.inmo.micro_utils.repos.exposed | ||||
|  | ||||
| import org.jetbrains.exposed.sql.Column | ||||
| import org.jetbrains.exposed.sql.SqlExpressionBuilder.eq | ||||
| import org.jetbrains.exposed.sql.SqlExpressionBuilder.isNotNull | ||||
| import org.jetbrains.exposed.sql.SqlExpressionBuilder.isNull | ||||
| import org.jetbrains.exposed.sql.SqlExpressionBuilder.neq | ||||
|  | ||||
| fun <T> Column<T?>.eqOrIsNull( | ||||
|     value: T? | ||||
| ) = if (value == null) { | ||||
|     isNull() | ||||
| } else { | ||||
|     eq(value) | ||||
| } | ||||
|  | ||||
| fun <T> Column<T?>.neqOrIsNotNull( | ||||
|     value: T? | ||||
| ) = if (value == null) { | ||||
|     isNotNull() | ||||
| } else { | ||||
|     neq(value) | ||||
| } | ||||
| @@ -1,12 +1,17 @@ | ||||
| package dev.inmo.micro_utils.repos.ktor.client.crud | ||||
|  | ||||
| import dev.inmo.micro_utils.ktor.client.createStandardWebsocketFlow | ||||
| import dev.inmo.micro_utils.ktor.common.* | ||||
| import dev.inmo.micro_utils.pagination.PaginationResult | ||||
| import dev.inmo.micro_utils.repos.* | ||||
| import dev.inmo.micro_utils.repos.ktor.common.crud.deletedObjectsIdsFlowRouting | ||||
| import dev.inmo.micro_utils.repos.ktor.common.crud.newObjectsFlowRouting | ||||
| import dev.inmo.micro_utils.repos.ktor.common.crud.updatedObjectsFlowRouting | ||||
| import io.ktor.client.HttpClient | ||||
| import io.ktor.http.ContentType | ||||
| import io.ktor.util.reflect.TypeInfo | ||||
| import io.ktor.util.reflect.typeInfo | ||||
| import kotlinx.coroutines.flow.Flow | ||||
| import kotlinx.serialization.* | ||||
|  | ||||
| class KtorCRUDRepoClient<ObjectType, IdType, InputValue> ( | ||||
| @@ -21,6 +26,15 @@ class KtorCRUDRepoClient<ObjectType, IdType, InputValue> ( | ||||
|             baseUrl: String, | ||||
|             httpClient: HttpClient, | ||||
|             contentType: ContentType, | ||||
|             newObjectsFlow: Flow<ObjectType> = httpClient.createStandardWebsocketFlow( | ||||
|                 buildStandardUrl(baseUrl, newObjectsFlowRouting), | ||||
|             ), | ||||
|             updatedObjectsFlow: Flow<ObjectType> = httpClient.createStandardWebsocketFlow( | ||||
|                 buildStandardUrl(baseUrl, updatedObjectsFlowRouting), | ||||
|             ), | ||||
|             deletedObjectsIdsFlow: Flow<IdType> = httpClient.createStandardWebsocketFlow( | ||||
|                 buildStandardUrl(baseUrl, deletedObjectsIdsFlowRouting), | ||||
|             ), | ||||
|             noinline idSerializer: suspend (IdType) -> String | ||||
|         ) = KtorCRUDRepoClient( | ||||
|             KtorReadCRUDRepoClient( | ||||
| @@ -35,7 +49,10 @@ class KtorCRUDRepoClient<ObjectType, IdType, InputValue> ( | ||||
|             KtorWriteCrudRepoClient<ObjectType, IdType, InputValue>( | ||||
|                 baseUrl, | ||||
|                 httpClient, | ||||
|                 contentType | ||||
|                 contentType, | ||||
|                 newObjectsFlow, | ||||
|                 updatedObjectsFlow, | ||||
|                 deletedObjectsIdsFlow | ||||
|             ) | ||||
|         ) | ||||
|  | ||||
| @@ -44,11 +61,23 @@ class KtorCRUDRepoClient<ObjectType, IdType, InputValue> ( | ||||
|             subpart: String, | ||||
|             httpClient: HttpClient, | ||||
|             contentType: ContentType, | ||||
|             newObjectsFlow: Flow<ObjectType> = httpClient.createStandardWebsocketFlow( | ||||
|                 buildStandardUrl(baseUrl, newObjectsFlowRouting), | ||||
|             ), | ||||
|             updatedObjectsFlow: Flow<ObjectType> = httpClient.createStandardWebsocketFlow( | ||||
|                 buildStandardUrl(baseUrl, updatedObjectsFlowRouting), | ||||
|             ), | ||||
|             deletedObjectsIdsFlow: Flow<IdType> = httpClient.createStandardWebsocketFlow( | ||||
|                 buildStandardUrl(baseUrl, deletedObjectsIdsFlowRouting), | ||||
|             ), | ||||
|             noinline idSerializer: suspend (IdType) -> String | ||||
|         ) = KtorCRUDRepoClient<ObjectType, IdType, InputValue>( | ||||
|             buildStandardUrl(baseUrl, subpart), | ||||
|             httpClient, | ||||
|             contentType, | ||||
|             newObjectsFlow, | ||||
|             updatedObjectsFlow, | ||||
|             deletedObjectsIdsFlow, | ||||
|             idSerializer | ||||
|         ) | ||||
|     } | ||||
| @@ -80,11 +109,23 @@ inline fun <reified ObjectType, reified IdType, reified InputValue> KtorCRUDRepo | ||||
|     subpart: String, | ||||
|     httpClient: HttpClient, | ||||
|     contentType: ContentType, | ||||
|     newObjectsFlow: Flow<ObjectType> = httpClient.createStandardWebsocketFlow( | ||||
|         buildStandardUrl(baseUrl, newObjectsFlowRouting), | ||||
|     ), | ||||
|     updatedObjectsFlow: Flow<ObjectType> = httpClient.createStandardWebsocketFlow( | ||||
|         buildStandardUrl(baseUrl, updatedObjectsFlowRouting), | ||||
|     ), | ||||
|     deletedObjectsIdsFlow: Flow<IdType> = httpClient.createStandardWebsocketFlow( | ||||
|         buildStandardUrl(baseUrl, deletedObjectsIdsFlowRouting), | ||||
|     ), | ||||
|     noinline idSerializer: suspend (IdType) -> String | ||||
| ) = KtorCRUDRepoClient<ObjectType, IdType, InputValue>( | ||||
|     buildStandardUrl(baseUrl, subpart), | ||||
|     httpClient, | ||||
|     contentType, | ||||
|     newObjectsFlow, | ||||
|     updatedObjectsFlow, | ||||
|     deletedObjectsIdsFlow, | ||||
|     idSerializer | ||||
| ) | ||||
|  | ||||
|   | ||||
| @@ -53,19 +53,22 @@ class KtorWriteCrudRepoClient<ObjectType, IdType, InputValue> ( | ||||
|         inline operator fun <reified ObjectType, reified IdType, reified InputValue> invoke( | ||||
|             baseUrl: String, | ||||
|             httpClient: HttpClient, | ||||
|             contentType: ContentType | ||||
|             contentType: ContentType, | ||||
|             newObjectsFlow: Flow<ObjectType> = httpClient.createStandardWebsocketFlow( | ||||
|                 buildStandardUrl(baseUrl, newObjectsFlowRouting), | ||||
|             ), | ||||
|             updatedObjectsFlow: Flow<ObjectType> = httpClient.createStandardWebsocketFlow( | ||||
|                 buildStandardUrl(baseUrl, updatedObjectsFlowRouting), | ||||
|             ), | ||||
|             deletedObjectsIdsFlow: Flow<IdType> = httpClient.createStandardWebsocketFlow( | ||||
|                 buildStandardUrl(baseUrl, deletedObjectsIdsFlowRouting), | ||||
|             ), | ||||
|         ) = KtorWriteCrudRepoClient<ObjectType, IdType, InputValue>( | ||||
|             baseUrl, | ||||
|             httpClient, | ||||
|             httpClient.createStandardWebsocketFlow( | ||||
|                 buildStandardUrl(baseUrl, newObjectsFlowRouting), | ||||
|             ), | ||||
|             httpClient.createStandardWebsocketFlow( | ||||
|                 buildStandardUrl(baseUrl, updatedObjectsFlowRouting), | ||||
|             ), | ||||
|             httpClient.createStandardWebsocketFlow( | ||||
|                 buildStandardUrl(baseUrl, deletedObjectsIdsFlowRouting), | ||||
|             ), | ||||
|             newObjectsFlow, | ||||
|             updatedObjectsFlow, | ||||
|             deletedObjectsIdsFlow, | ||||
|             { | ||||
|                 contentType(contentType) | ||||
|                 setBody(it) | ||||
|   | ||||
| @@ -1,10 +1,14 @@ | ||||
| package dev.inmo.micro_utils.repos.ktor.client.key.value | ||||
|  | ||||
| import dev.inmo.micro_utils.ktor.client.createStandardWebsocketFlow | ||||
| import dev.inmo.micro_utils.ktor.common.* | ||||
| import dev.inmo.micro_utils.repos.* | ||||
| import dev.inmo.micro_utils.repos.ktor.common.key_value.onNewValueRoute | ||||
| import dev.inmo.micro_utils.repos.ktor.common.key_value.onValueRemovedRoute | ||||
| import io.ktor.client.HttpClient | ||||
| import io.ktor.http.ContentType | ||||
| import io.ktor.http.encodeURLQueryComponent | ||||
| import kotlinx.coroutines.flow.Flow | ||||
| import kotlinx.serialization.* | ||||
|  | ||||
| class KtorKeyValueRepoClient<Key, Value> ( | ||||
| @@ -20,6 +24,12 @@ class KtorKeyValueRepoClient<Key, Value> ( | ||||
|             httpClient: HttpClient, | ||||
|             contentType: ContentType, | ||||
|             noinline idSerializer: suspend (Key) -> String, | ||||
|             onNewValue: Flow<Pair<Key, Value>> = httpClient.createStandardWebsocketFlow( | ||||
|                 buildStandardUrl(baseUrl, onNewValueRoute), | ||||
|             ), | ||||
|             onValueRemoved: Flow<Key> = httpClient.createStandardWebsocketFlow( | ||||
|                 buildStandardUrl(baseUrl, onValueRemovedRoute), | ||||
|             ), | ||||
|             noinline valueSerializer: suspend (Value) -> String | ||||
|         ) = KtorKeyValueRepoClient( | ||||
|             KtorReadKeyValueRepoClient( | ||||
| @@ -28,7 +38,9 @@ class KtorKeyValueRepoClient<Key, Value> ( | ||||
|             KtorWriteKeyValueRepoClient( | ||||
|                 baseUrl, | ||||
|                 httpClient, | ||||
|                 contentType | ||||
|                 contentType, | ||||
|                 onNewValue, | ||||
|                 onValueRemoved | ||||
|             ) | ||||
|         ) | ||||
|         inline operator fun <reified Key, reified Value> invoke( | ||||
| @@ -37,12 +49,20 @@ class KtorKeyValueRepoClient<Key, Value> ( | ||||
|             httpClient: HttpClient, | ||||
|             contentType: ContentType, | ||||
|             noinline idSerializer: suspend (Key) -> String, | ||||
|             onNewValue: Flow<Pair<Key, Value>> = httpClient.createStandardWebsocketFlow( | ||||
|                 buildStandardUrl(baseUrl, onNewValueRoute), | ||||
|             ), | ||||
|             onValueRemoved: Flow<Key> = httpClient.createStandardWebsocketFlow( | ||||
|                 buildStandardUrl(baseUrl, onValueRemovedRoute), | ||||
|             ), | ||||
|             noinline valueSerializer: suspend (Value) -> String | ||||
|         ) = KtorKeyValueRepoClient( | ||||
|             buildStandardUrl(baseUrl, subpart), | ||||
|             httpClient, | ||||
|             contentType, | ||||
|             idSerializer, | ||||
|             onNewValue, | ||||
|             onValueRemoved, | ||||
|             valueSerializer | ||||
|         ) | ||||
|     } | ||||
|   | ||||
| @@ -60,17 +60,19 @@ class KtorWriteKeyValueRepoClient<Key, Value>( | ||||
|         inline operator fun <reified Key, reified Value> invoke( | ||||
|             baseUrl: String, | ||||
|             httpClient: HttpClient, | ||||
|             contentType: ContentType | ||||
|             contentType: ContentType, | ||||
|             onNewValue: Flow<Pair<Key, Value>> = httpClient.createStandardWebsocketFlow( | ||||
|                 buildStandardUrl(baseUrl, onNewValueRoute), | ||||
|             ), | ||||
|             onValueRemoved: Flow<Key> = httpClient.createStandardWebsocketFlow( | ||||
|                 buildStandardUrl(baseUrl, onValueRemovedRoute), | ||||
|             ), | ||||
|         ) = KtorWriteKeyValueRepoClient<Key, Value>( | ||||
|             baseUrl, | ||||
|             httpClient, | ||||
|             contentType, | ||||
|             httpClient.createStandardWebsocketFlow( | ||||
|                 buildStandardUrl(baseUrl, onNewValueRoute), | ||||
|             ), | ||||
|             httpClient.createStandardWebsocketFlow( | ||||
|                 buildStandardUrl(baseUrl, onValueRemovedRoute), | ||||
|             ), | ||||
|             onNewValue, | ||||
|             onValueRemoved, | ||||
|             typeInfo<List<Key>>(), | ||||
|             typeInfo<List<Value>>(), | ||||
|             typeInfo<Map<Key, Value>>() | ||||
|   | ||||
| @@ -1,10 +1,15 @@ | ||||
| package dev.inmo.micro_utils.repos.ktor.client.key.values | ||||
|  | ||||
| import dev.inmo.micro_utils.ktor.client.createStandardWebsocketFlow | ||||
| import dev.inmo.micro_utils.ktor.common.* | ||||
| import dev.inmo.micro_utils.repos.* | ||||
| import dev.inmo.micro_utils.repos.ktor.common.one_to_many.onDataClearedRoute | ||||
| import dev.inmo.micro_utils.repos.ktor.common.one_to_many.onNewValueRoute | ||||
| import dev.inmo.micro_utils.repos.ktor.common.one_to_many.onValueRemovedRoute | ||||
| import io.ktor.client.HttpClient | ||||
| import io.ktor.http.ContentType | ||||
| import io.ktor.http.encodeURLQueryComponent | ||||
| import kotlinx.coroutines.flow.Flow | ||||
| import kotlinx.serialization.* | ||||
|  | ||||
| class KtorKeyValuesRepoClient<Key, Value> ( | ||||
| @@ -20,6 +25,15 @@ class KtorKeyValuesRepoClient<Key, Value> ( | ||||
|             httpClient: HttpClient, | ||||
|             contentType: ContentType, | ||||
|             noinline keySerializer: suspend (Key) -> String, | ||||
|             onNewValue: Flow<Pair<Key, Value>> = httpClient.createStandardWebsocketFlow( | ||||
|                 buildStandardUrl(baseUrl, onNewValueRoute), | ||||
|             ), | ||||
|             onValueRemoved: Flow<Pair<Key, Value>> = httpClient.createStandardWebsocketFlow( | ||||
|                 buildStandardUrl(baseUrl, onValueRemovedRoute), | ||||
|             ), | ||||
|             onDataCleared: Flow<Key> = httpClient.createStandardWebsocketFlow( | ||||
|                 buildStandardUrl(baseUrl, onDataClearedRoute), | ||||
|             ), | ||||
|             noinline valueSerializer: suspend (Value) -> String | ||||
|         ) = KtorKeyValuesRepoClient( | ||||
|             KtorReadKeyValuesRepoClient( | ||||
| @@ -32,7 +46,10 @@ class KtorKeyValuesRepoClient<Key, Value> ( | ||||
|             KtorWriteKeyValuesRepoClient( | ||||
|                 baseUrl, | ||||
|                 httpClient, | ||||
|                 contentType | ||||
|                 contentType, | ||||
|                 onNewValue, | ||||
|                 onValueRemoved, | ||||
|                 onDataCleared | ||||
|             ) | ||||
|         ) | ||||
|         inline operator fun <reified Key : Any, reified Value : Any> invoke( | ||||
| @@ -41,12 +58,24 @@ class KtorKeyValuesRepoClient<Key, Value> ( | ||||
|             httpClient: HttpClient, | ||||
|             contentType: ContentType, | ||||
|             noinline keySerializer: suspend (Key) -> String, | ||||
|             onNewValue: Flow<Pair<Key, Value>> = httpClient.createStandardWebsocketFlow( | ||||
|                 buildStandardUrl(baseUrl, onNewValueRoute), | ||||
|             ), | ||||
|             onValueRemoved: Flow<Pair<Key, Value>> = httpClient.createStandardWebsocketFlow( | ||||
|                 buildStandardUrl(baseUrl, onValueRemovedRoute), | ||||
|             ), | ||||
|             onDataCleared: Flow<Key> = httpClient.createStandardWebsocketFlow( | ||||
|                 buildStandardUrl(baseUrl, onDataClearedRoute), | ||||
|             ), | ||||
|             noinline valueSerializer: suspend (Value) -> String | ||||
|         ) = KtorKeyValuesRepoClient( | ||||
|             buildStandardUrl(baseUrl, subpart), | ||||
|             httpClient, | ||||
|             contentType, | ||||
|             keySerializer, | ||||
|             onNewValue, | ||||
|             onValueRemoved, | ||||
|             onDataCleared, | ||||
|             valueSerializer | ||||
|         ) | ||||
|     } | ||||
| @@ -59,13 +88,25 @@ inline fun <reified Key : Any, reified Value : Any> KtorKeyValuesRepoClient( | ||||
|     keySerializer: SerializationStrategy<Key>, | ||||
|     valueSerializer: SerializationStrategy<Value>, | ||||
|     serialFormat: StringFormat, | ||||
|     onNewValue: Flow<Pair<Key, Value>> = httpClient.createStandardWebsocketFlow( | ||||
|         buildStandardUrl(baseUrl, onNewValueRoute), | ||||
|     ), | ||||
|     onValueRemoved: Flow<Pair<Key, Value>> = httpClient.createStandardWebsocketFlow( | ||||
|         buildStandardUrl(baseUrl, onValueRemovedRoute), | ||||
|     ), | ||||
|     onDataCleared: Flow<Key> = httpClient.createStandardWebsocketFlow( | ||||
|         buildStandardUrl(baseUrl, onDataClearedRoute), | ||||
|     ), | ||||
| ) = KtorKeyValuesRepoClient<Key, Value>( | ||||
|     baseUrl, | ||||
|     httpClient, | ||||
|     contentType, | ||||
|     { | ||||
|         serialFormat.encodeToString(keySerializer, it).encodeURLQueryComponent() | ||||
|     } | ||||
|     }, | ||||
|     onNewValue, | ||||
|     onValueRemoved, | ||||
|     onDataCleared | ||||
| ) { | ||||
|     serialFormat.encodeToString(valueSerializer, it).encodeURLQueryComponent() | ||||
| } | ||||
| @@ -77,13 +118,25 @@ inline fun <reified Key : Any, reified Value : Any> KtorKeyValuesRepoClient( | ||||
|     keySerializer: SerializationStrategy<Key>, | ||||
|     valueSerializer: SerializationStrategy<Value>, | ||||
|     serialFormat: BinaryFormat, | ||||
|     onNewValue: Flow<Pair<Key, Value>> = httpClient.createStandardWebsocketFlow( | ||||
|         buildStandardUrl(baseUrl, onNewValueRoute), | ||||
|     ), | ||||
|     onValueRemoved: Flow<Pair<Key, Value>> = httpClient.createStandardWebsocketFlow( | ||||
|         buildStandardUrl(baseUrl, onValueRemovedRoute), | ||||
|     ), | ||||
|     onDataCleared: Flow<Key> = httpClient.createStandardWebsocketFlow( | ||||
|         buildStandardUrl(baseUrl, onDataClearedRoute), | ||||
|     ), | ||||
| ) = KtorKeyValuesRepoClient<Key, Value>( | ||||
|     baseUrl, | ||||
|     httpClient, | ||||
|     contentType, | ||||
|     { | ||||
|         serialFormat.encodeHex(keySerializer, it) | ||||
|     } | ||||
|     }, | ||||
|     onNewValue, | ||||
|     onValueRemoved, | ||||
|     onDataCleared | ||||
| ) { | ||||
|     serialFormat.encodeHex(valueSerializer, it) | ||||
| } | ||||
|   | ||||
| @@ -84,20 +84,23 @@ class KtorWriteKeyValuesRepoClient<Key : Any, Value : Any>( | ||||
|         inline operator fun <reified Key : Any, reified Value : Any> invoke( | ||||
|             baseUrl: String, | ||||
|             httpClient: HttpClient, | ||||
|             contentType: ContentType | ||||
|             contentType: ContentType, | ||||
|             onNewValue: Flow<Pair<Key, Value>> = httpClient.createStandardWebsocketFlow( | ||||
|                 buildStandardUrl(baseUrl, onNewValueRoute), | ||||
|             ), | ||||
|             onValueRemoved: Flow<Pair<Key, Value>> = httpClient.createStandardWebsocketFlow( | ||||
|                 buildStandardUrl(baseUrl, onValueRemovedRoute), | ||||
|             ), | ||||
|             onDataCleared: Flow<Key> = httpClient.createStandardWebsocketFlow( | ||||
|                 buildStandardUrl(baseUrl, onDataClearedRoute), | ||||
|             ), | ||||
|         ) = KtorWriteKeyValuesRepoClient<Key, Value>( | ||||
|             baseUrl, | ||||
|             httpClient, | ||||
|             contentType, | ||||
|             httpClient.createStandardWebsocketFlow( | ||||
|                 buildStandardUrl(baseUrl, onNewValueRoute), | ||||
|             ), | ||||
|             httpClient.createStandardWebsocketFlow( | ||||
|                 buildStandardUrl(baseUrl, onValueRemovedRoute), | ||||
|             ), | ||||
|             httpClient.createStandardWebsocketFlow( | ||||
|                 buildStandardUrl(baseUrl, onDataClearedRoute), | ||||
|             ), | ||||
|             onNewValue, | ||||
|             onValueRemoved, | ||||
|             onDataCleared, | ||||
|             typeInfo<Key>(), | ||||
|             typeInfo<Value>(), | ||||
|             typeInfo<Map<Key, List<Value>>>() | ||||
|   | ||||
| @@ -14,6 +14,7 @@ import kotlinx.coroutines.launch | ||||
| import kotlinx.serialization.SerialFormat | ||||
| import kotlinx.serialization.StringFormat | ||||
| import kotlinx.serialization.json.JsonObject | ||||
| import kotlinx.serialization.json.decodeFromJsonElement | ||||
| import kotlinx.serialization.json.jsonObject | ||||
| import org.koin.core.Koin | ||||
| import org.koin.core.KoinApplication | ||||
| @@ -93,18 +94,19 @@ object StartLauncherPlugin : StartPlugin { | ||||
|  | ||||
|     /** | ||||
|      * Will create [KoinApplication], init, load modules using [StartLauncherPlugin] and start plugins using the same base | ||||
|      * plugin | ||||
|      * plugin. It is basic [start] method which accepts both [config] and [rawConfig] which suppose to be the same or | ||||
|      * at least [rawConfig] must contain serialized variant of [config] | ||||
|      * | ||||
|      * @param rawConfig It is expected that this [JsonObject] will contain serialized [Config] ([StartLauncherPlugin] will | ||||
|      * deserialize it in its [StartLauncherPlugin.setupDI] | ||||
|      */ | ||||
|     suspend fun start(rawConfig: JsonObject) { | ||||
|     suspend fun start(config: Config, rawConfig: JsonObject) { | ||||
|  | ||||
|         logger.i("Start initialization") | ||||
|         val koinApp = KoinApplication.init() | ||||
|         koinApp.modules( | ||||
|             module { | ||||
|                 setupDI(rawConfig) | ||||
|                 setupDI(config, rawConfig) | ||||
|             } | ||||
|         ) | ||||
|         logger.i("Modules loaded") | ||||
| @@ -116,26 +118,26 @@ object StartLauncherPlugin : StartPlugin { | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Will create [KoinApplication], init, load modules using [StartLauncherPlugin] and start plugins using the same base | ||||
|      * plugin | ||||
|      * Call [start] with deserialized [Config] as config and [rawConfig] as is | ||||
|      * | ||||
|      * @param config In difference with other [start] method here config is already deserialized and this config will | ||||
|      * be converted to [JsonObject] as raw config. That means that all plugins from [config] will receive | ||||
|      * serialized version of [config] in [StartPlugin.setupDI] method | ||||
|      * @param rawConfig It is expected that this [JsonObject] will contain serialized [Config] | ||||
|      */ | ||||
|     suspend fun start(rawConfig: JsonObject) { | ||||
|  | ||||
|         start(defaultJson.decodeFromJsonElement(Config.serializer(), rawConfig), rawConfig) | ||||
|  | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Call [start] with deserialized [Config] as is and serialize it to [JsonObject] to pass as the first parameter | ||||
|      * to the basic [start] method | ||||
|      * | ||||
|      * @param config Will be converted to [JsonObject] as raw config. That means that all plugins from [config] will | ||||
|      * receive serialized version of [config] in [StartPlugin.setupDI] method | ||||
|      */ | ||||
|     suspend fun start(config: Config) { | ||||
|  | ||||
|         logger.i("Start initialization") | ||||
|         val koinApp = KoinApplication.init() | ||||
|         logger.i("Koin app created") | ||||
|         koinApp.modules( | ||||
|             module { | ||||
|                 setupDI(config) | ||||
|             } | ||||
|         ) | ||||
|         startKoin(koinApp) | ||||
|         logger.i("Koin started") | ||||
|         startPlugin(koinApp.koin) | ||||
|         start(config, defaultJson.encodeToJsonElement(Config.serializer(), config).jsonObject) | ||||
|  | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| package dev.inmo.micro_utils.startup.launcher | ||||
|  | ||||
| import dev.inmo.kslog.common.KSLog | ||||
| import dev.inmo.kslog.common.LogLevel | ||||
| import dev.inmo.kslog.common.i | ||||
| import kotlinx.serialization.json.jsonObject | ||||
| import java.io.File | ||||
| @@ -23,10 +24,25 @@ import java.io.File | ||||
|  * In that case in `build/distributions` folder you will be able to find zip and tar files with all required | ||||
|  * tools for application running (via their `bin/app_name` binary). In that case yoy will not need to pass | ||||
|  * `--args=...` and launch will look like `./bin/app_name sample.config.json` | ||||
|  * | ||||
|  * ## Debug mode | ||||
|  * | ||||
|  * You may pass the second parameter, one of [LogLevel] enum variants to setup [KSLog] minimal logging level. Sample: | ||||
|  * | ||||
|  * ```bash | ||||
|  * ./gradlew run --args="sample.config.json DEBUG" // enable debugging output | ||||
|  * ``` | ||||
|  * | ||||
|  * OR | ||||
|  * ```bash | ||||
|  * ./gradlew run --args="sample.config.json WARNING" // enable logging since WARNING | ||||
|  * ``` | ||||
|  * | ||||
|  * **Default level is [LogLevel.INFO]** | ||||
|  */ | ||||
| suspend fun main(args: Array<String>) { | ||||
|  | ||||
|     KSLog.default = KSLog("Launcher") | ||||
|     KSLog.default = KSLog("Launcher", args.getOrNull(1) ?.let { LogLevel.valueOf(it) } ?: LogLevel.INFO) | ||||
|     val (configPath) = args | ||||
|     val file = File(configPath) | ||||
|     KSLog.i("Start read config from ${file.absolutePath}") | ||||
|   | ||||
							
								
								
									
										10
									
								
								startup/template/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								startup/template/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | ||||
| # How to use | ||||
|  | ||||
| In case you have multiplatform project and wish to use startup plugin, this template may help you to create new modules. | ||||
|  | ||||
| 1. Copy-paste whole template folder (you may clone this folder to your project and actualize some data to copy your prepared template) | ||||
| 2. Replace `group_name` by your project (or root module) group name | ||||
| 3. Replace `module_name` by the name of your new module name | ||||
|  | ||||
| You may read about the `build.gradle` structure in these templates in project | ||||
| [KotlinMultiplatformProjectTemplate](https://github.com/InsanusMokrassar/KotlinMultiplatformProjectTemplate). | ||||
							
								
								
									
										18
									
								
								startup/template/client/build.gradle
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								startup/template/client/build.gradle
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| plugins { | ||||
|     id "org.jetbrains.kotlin.multiplatform" | ||||
|     id "org.jetbrains.kotlin.plugin.serialization" | ||||
|     id "com.android.library" | ||||
|     alias(libs.plugins.compose) | ||||
| } | ||||
|  | ||||
| apply from: "$mppProjectWithSerializationPresetPath" | ||||
|  | ||||
| kotlin { | ||||
|     sourceSets { | ||||
|         commonMain { | ||||
|             dependencies { | ||||
|                 api project(":${rootProject.name}.module_name.common") | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,15 @@ | ||||
| package group_name.module_name.client | ||||
|  | ||||
| import dev.inmo.micro_utils.startup.plugin.StartPlugin | ||||
| import kotlinx.serialization.json.JsonObject | ||||
| import org.koin.core.Koin | ||||
| import org.koin.core.module.Module | ||||
|  | ||||
| object ClientPlugin : StartPlugin { | ||||
|     override fun Module.setupDI(config: JsonObject) { | ||||
|     } | ||||
|  | ||||
|     override suspend fun startPlugin(koin: Koin) { | ||||
|         super.startPlugin(koin) | ||||
|     } | ||||
| } | ||||
							
								
								
									
										20
									
								
								startup/template/client/src/jsMain/kotlin/ClientJSPlugin.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								startup/template/client/src/jsMain/kotlin/ClientJSPlugin.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | ||||
| package group_name.module_name.client | ||||
|  | ||||
| import group_name.module_name.common.CommonJSPlugin | ||||
| import dev.inmo.micro_utils.startup.plugin.StartPlugin | ||||
| import kotlinx.serialization.json.JsonObject | ||||
| import org.koin.core.Koin | ||||
| import org.koin.core.module.Module | ||||
|  | ||||
| object ClientJSPlugin : StartPlugin { | ||||
|     override fun Module.setupDI(config: JsonObject) { | ||||
|         with(CommonJSPlugin) { setupDI(config) } | ||||
|         with(ClientPlugin) { setupDI(config) } | ||||
|     } | ||||
|  | ||||
|     override suspend fun startPlugin(koin: Koin) { | ||||
|         super.startPlugin(koin) | ||||
|         CommonJSPlugin.startPlugin(koin) | ||||
|         ClientPlugin.startPlugin(koin) | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,9 @@ | ||||
| package group_name.module_name.client | ||||
|  | ||||
| import dev.inmo.micro_utils.startup.plugin.createStartupPluginAndRegister | ||||
|  | ||||
| @ExperimentalStdlibApi | ||||
| @EagerInitialization | ||||
| @JsExport | ||||
| @ExperimentalJsExport | ||||
| private val jsModuleLoader = createStartupPluginAndRegister("template.ClientJSPlugin") { ClientJSPlugin } | ||||
| @@ -0,0 +1,21 @@ | ||||
| package group_name.module_name.client | ||||
|  | ||||
| import group_name.module_name.common.CommonJVMPlugin | ||||
| import group_name.module_name.common.CommonJVMPlugin.setupDI | ||||
| import dev.inmo.micro_utils.startup.plugin.StartPlugin | ||||
| import kotlinx.serialization.json.JsonObject | ||||
| import org.koin.core.Koin | ||||
| import org.koin.core.module.Module | ||||
|  | ||||
| object ClientJVMPlugin : StartPlugin { | ||||
|     override fun Module.setupDI(config: JsonObject) { | ||||
|         with(CommonJVMPlugin) { setupDI(config) } | ||||
|         with(ClientPlugin) { setupDI(config) } | ||||
|     } | ||||
|  | ||||
|     override suspend fun startPlugin(koin: Koin) { | ||||
|         super.startPlugin(koin) | ||||
|         CommonJVMPlugin.startPlugin(koin) | ||||
|         ClientPlugin.startPlugin(koin) | ||||
|     } | ||||
| } | ||||
							
								
								
									
										1
									
								
								startup/template/client/src/main/AndroidManifest.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								startup/template/client/src/main/AndroidManifest.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| <manifest package="group_name.module_name.client"/> | ||||
| @@ -0,0 +1,21 @@ | ||||
| package group_name.module_name.client | ||||
|  | ||||
| import group_name.module_name.common.CommonAndroidPlugin | ||||
| import group_name.module_name.common.CommonAndroidPlugin.setupDI | ||||
| import dev.inmo.micro_utils.startup.plugin.StartPlugin | ||||
| import kotlinx.serialization.json.JsonObject | ||||
| import org.koin.core.Koin | ||||
| import org.koin.core.module.Module | ||||
|  | ||||
| object ClientAndroidPlugin : StartPlugin { | ||||
|     override fun Module.setupDI(config: JsonObject) { | ||||
|         with(CommonAndroidPlugin) { setupDI(config) } | ||||
|         with(ClientPlugin) { setupDI(config) } | ||||
|     } | ||||
|  | ||||
|     override suspend fun startPlugin(koin: Koin) { | ||||
|         super.startPlugin(koin) | ||||
|         CommonAndroidPlugin.startPlugin(koin) | ||||
|         ClientPlugin.startPlugin(koin) | ||||
|     } | ||||
| } | ||||
							
								
								
									
										7
									
								
								startup/template/common/build.gradle
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								startup/template/common/build.gradle
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | ||||
| plugins { | ||||
|     id "org.jetbrains.kotlin.multiplatform" | ||||
|     id "org.jetbrains.kotlin.plugin.serialization" | ||||
|     id "com.android.library" | ||||
| } | ||||
|  | ||||
| apply from: "$mppProjectWithSerializationPresetPath" | ||||
| @@ -0,0 +1,11 @@ | ||||
| package group_name.module_name.common | ||||
|  | ||||
| import dev.inmo.micro_utils.startup.plugin.StartPlugin | ||||
| import kotlinx.serialization.json.JsonObject | ||||
| import org.koin.core.module.Module | ||||
|  | ||||
| object CommonPlugin : StartPlugin { | ||||
|     override fun Module.setupDI(config: JsonObject) { | ||||
|  | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1 @@ | ||||
| package group_name.module_name.common | ||||
							
								
								
									
										17
									
								
								startup/template/common/src/jsMain/kotlin/CommonJSPlugin.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								startup/template/common/src/jsMain/kotlin/CommonJSPlugin.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | ||||
| package group_name.module_name.common | ||||
|  | ||||
| import dev.inmo.micro_utils.startup.plugin.StartPlugin | ||||
| import kotlinx.serialization.json.JsonObject | ||||
| import org.koin.core.Koin | ||||
| import org.koin.core.module.Module | ||||
|  | ||||
| object CommonJSPlugin : StartPlugin { | ||||
|     override fun Module.setupDI(config: JsonObject) { | ||||
|         with (CommonPlugin) { setupDI(config) } | ||||
|     } | ||||
|  | ||||
|     override suspend fun startPlugin(koin: Koin) { | ||||
|         super.startPlugin(koin) | ||||
|         CommonPlugin.startPlugin(koin) | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,9 @@ | ||||
| package group_name.module_name.common | ||||
|  | ||||
| import dev.inmo.micro_utils.startup.plugin.createStartupPluginAndRegister | ||||
|  | ||||
| @ExperimentalStdlibApi | ||||
| @EagerInitialization | ||||
| @JsExport | ||||
| @ExperimentalJsExport | ||||
| private val jsModuleLoader = createStartupPluginAndRegister("template.CommonJSPlugin") { CommonJSPlugin } | ||||
| @@ -0,0 +1,17 @@ | ||||
| package group_name.module_name.common | ||||
|  | ||||
| import dev.inmo.micro_utils.startup.plugin.StartPlugin | ||||
| import kotlinx.serialization.json.JsonObject | ||||
| import org.koin.core.Koin | ||||
| import org.koin.core.module.Module | ||||
|  | ||||
| object CommonJVMPlugin : StartPlugin { | ||||
|     override fun Module.setupDI(config: JsonObject) { | ||||
|         with (CommonPlugin) { setupDI(config) } | ||||
|     } | ||||
|  | ||||
|     override suspend fun startPlugin(koin: Koin) { | ||||
|         super.startPlugin(koin) | ||||
|         CommonPlugin.startPlugin(koin) | ||||
|     } | ||||
| } | ||||
							
								
								
									
										1
									
								
								startup/template/common/src/main/AndroidManifest.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								startup/template/common/src/main/AndroidManifest.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| <manifest package="group_name.module_name.common"/> | ||||
| @@ -0,0 +1,17 @@ | ||||
| package group_name.module_name.common | ||||
|  | ||||
| import dev.inmo.micro_utils.startup.plugin.StartPlugin | ||||
| import kotlinx.serialization.json.JsonObject | ||||
| import org.koin.core.Koin | ||||
| import org.koin.core.module.Module | ||||
|  | ||||
| object CommonAndroidPlugin : StartPlugin { | ||||
|     override fun Module.setupDI(config: JsonObject) { | ||||
|         with (CommonPlugin) { setupDI(config) } | ||||
|     } | ||||
|  | ||||
|     override suspend fun startPlugin(koin: Koin) { | ||||
|         super.startPlugin(koin) | ||||
|         CommonPlugin.startPlugin(koin) | ||||
|     } | ||||
| } | ||||
							
								
								
									
										16
									
								
								startup/template/server/build.gradle
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								startup/template/server/build.gradle
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | ||||
| plugins { | ||||
|     id "org.jetbrains.kotlin.multiplatform" | ||||
|     id "org.jetbrains.kotlin.plugin.serialization" | ||||
| } | ||||
|  | ||||
| apply from: "$mppJavaProjectPresetPath" | ||||
|  | ||||
| kotlin { | ||||
|     sourceSets { | ||||
|         commonMain { | ||||
|             dependencies { | ||||
|                 api project(":${rootProject.name}.module_name.common") | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,15 @@ | ||||
| package group_name.module_name.server | ||||
|  | ||||
| import dev.inmo.micro_utils.startup.plugin.StartPlugin | ||||
| import kotlinx.serialization.json.JsonObject | ||||
| import org.koin.core.Koin | ||||
| import org.koin.core.module.Module | ||||
|  | ||||
| object ServerPlugin : StartPlugin { | ||||
|     override fun Module.setupDI(config: JsonObject) { | ||||
|     } | ||||
|  | ||||
|     override suspend fun startPlugin(koin: Koin) { | ||||
|         super.startPlugin(koin) | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,20 @@ | ||||
| package group_name.module_name.server | ||||
|  | ||||
| import group_name.module_name.common.CommonJVMPlugin | ||||
| import dev.inmo.micro_utils.startup.plugin.StartPlugin | ||||
| import kotlinx.serialization.json.JsonObject | ||||
| import org.koin.core.Koin | ||||
| import org.koin.core.module.Module | ||||
|  | ||||
| object ServerJVMPlugin : StartPlugin { | ||||
|     override fun Module.setupDI(config: JsonObject) { | ||||
|         with(CommonJVMPlugin) { setupDI(config) } | ||||
|         with(ServerPlugin) { setupDI(config) } | ||||
|     } | ||||
|  | ||||
|     override suspend fun startPlugin(koin: Koin) { | ||||
|         super.startPlugin(koin) | ||||
|         CommonJVMPlugin.startPlugin(koin) | ||||
|         ServerPlugin.startPlugin(koin) | ||||
|     } | ||||
| } | ||||
		Reference in New Issue
	
	Block a user