mirror of
https://github.com/InsanusMokrassar/MicroUtils.git
synced 2024-12-23 09:07:14 +00:00
commit
48b816aa22
10
CHANGELOG.md
10
CHANGELOG.md
@ -1,5 +1,15 @@
|
||||
# Changelog
|
||||
|
||||
## 0.20.36
|
||||
|
||||
* `Versions`:
|
||||
* `Serialization`: `1.6.2` -> `1.6.3`
|
||||
* `Korlibs`: `5.3.1` -> `5.3.2`
|
||||
* `Repos`:
|
||||
* `Cache`:
|
||||
* Improve work and functionality of `actualizeAll` and subsequent functions
|
||||
* All internal repos `invalidate`/`actualizeAll` now use common `actualizeAll` functions
|
||||
|
||||
## 0.20.35
|
||||
|
||||
* `Versions`:
|
||||
|
@ -15,5 +15,5 @@ crypto_js_version=4.1.1
|
||||
# Project data
|
||||
|
||||
group=dev.inmo
|
||||
version=0.20.35
|
||||
android_code_version=241
|
||||
version=0.20.36
|
||||
android_code_version=242
|
||||
|
@ -1,7 +1,7 @@
|
||||
[versions]
|
||||
|
||||
kt = "1.9.22"
|
||||
kt-serialization = "1.6.2"
|
||||
kt-serialization = "1.6.3"
|
||||
kt-coroutines = "1.8.0"
|
||||
|
||||
kslog = "1.3.2"
|
||||
@ -10,7 +10,7 @@ jb-compose = "1.5.12"
|
||||
jb-exposed = "0.47.0"
|
||||
jb-dokka = "1.9.10"
|
||||
|
||||
korlibs = "5.3.1"
|
||||
korlibs = "5.3.2"
|
||||
uuid = "0.8.2"
|
||||
|
||||
ktor = "2.3.8"
|
||||
|
@ -5,6 +5,7 @@ import dev.inmo.micro_utils.coroutines.withReadAcquire
|
||||
import dev.inmo.micro_utils.coroutines.withWriteLock
|
||||
import dev.inmo.micro_utils.repos.*
|
||||
import dev.inmo.micro_utils.repos.cache.cache.KVCache
|
||||
import dev.inmo.micro_utils.repos.cache.util.ActualizeAllClearMode
|
||||
import dev.inmo.micro_utils.repos.cache.util.actualizeAll
|
||||
import kotlinx.coroutines.CoroutineScope
|
||||
import kotlinx.coroutines.Dispatchers
|
||||
@ -29,7 +30,7 @@ open class ReadCRUDCacheRepo<ObjectType, IdType>(
|
||||
kvCache.getAll()
|
||||
}.takeIf { it.size.toLong() == count() } ?: parentRepo.getAll().also {
|
||||
locker.withWriteLock {
|
||||
kvCache.actualizeAll(true) { it }
|
||||
kvCache.actualizeAll(clearMode = ActualizeAllClearMode.BeforeSet) { it }
|
||||
}
|
||||
}
|
||||
}
|
||||
@ -148,7 +149,9 @@ open class CRUDCacheRepo<ObjectType, IdType, InputValueType>(
|
||||
locker,
|
||||
idGetter
|
||||
),
|
||||
CRUDRepo<ObjectType, IdType, InputValueType>
|
||||
CRUDRepo<ObjectType, IdType, InputValueType> {
|
||||
override suspend fun invalidate() = kvCache.actualizeAll(parentRepo, locker = locker)
|
||||
}
|
||||
|
||||
fun <ObjectType, IdType, InputType> CRUDRepo<ObjectType, IdType, InputType>.cached(
|
||||
kvCache: KVCache<IdType, ObjectType>,
|
||||
|
@ -1,5 +1,11 @@
|
||||
package dev.inmo.micro_utils.repos.cache
|
||||
|
||||
interface CacheRepo {
|
||||
interface InvalidatableRepo {
|
||||
/**
|
||||
* Invalidates its internal data. It __may__ lead to autoreload of data. In case when repo makes autoreload,
|
||||
* it must do loading of data __before__ clear
|
||||
*/
|
||||
suspend fun invalidate()
|
||||
}
|
||||
|
||||
typealias CacheRepo = InvalidatableRepo
|
||||
|
@ -6,6 +6,7 @@ import dev.inmo.micro_utils.coroutines.withWriteLock
|
||||
import dev.inmo.micro_utils.pagination.*
|
||||
import dev.inmo.micro_utils.repos.*
|
||||
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
|
||||
import kotlinx.coroutines.flow.*
|
||||
@ -48,9 +49,7 @@ open class ReadKeyValueCacheRepo<Key,Value>(
|
||||
}
|
||||
}
|
||||
|
||||
override suspend fun invalidate() = locker.withWriteLock {
|
||||
kvCache.clear()
|
||||
}
|
||||
override suspend fun invalidate() = kvCache.actualizeAll(parentRepo, locker = locker)
|
||||
}
|
||||
|
||||
fun <Key, Value> ReadKeyValueRepo<Key, Value>.cached(
|
||||
@ -75,10 +74,6 @@ open class KeyValueCacheRepo<Key,Value>(
|
||||
}
|
||||
}.launchIn(scope)
|
||||
|
||||
override suspend fun invalidate() = locker.withWriteLock {
|
||||
kvCache.clear()
|
||||
}
|
||||
|
||||
override suspend fun clear() {
|
||||
parentRepo.clear()
|
||||
locker.withWriteLock {
|
||||
|
@ -7,6 +7,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.KVCache
|
||||
import dev.inmo.micro_utils.repos.cache.util.actualizeAll
|
||||
import kotlinx.coroutines.CoroutineScope
|
||||
import kotlinx.coroutines.Dispatchers
|
||||
import kotlinx.coroutines.flow.*
|
||||
@ -47,9 +48,7 @@ open class ReadKeyValuesCacheRepo<Key,Value>(
|
||||
kvCache.contains(k)
|
||||
} || parentRepo.contains(k)
|
||||
|
||||
override suspend fun invalidate() = locker.withWriteLock {
|
||||
kvCache.clear()
|
||||
}
|
||||
override suspend fun invalidate() = kvCache.actualizeAll(parentRepo, locker = locker)
|
||||
}
|
||||
|
||||
fun <Key, Value> ReadKeyValuesRepo<Key, Value>.cached(
|
||||
@ -84,10 +83,6 @@ open class KeyValuesCacheRepo<Key,Value>(
|
||||
kvCache.unset(it)
|
||||
}
|
||||
}.launchIn(scope)
|
||||
|
||||
override suspend fun invalidate() = locker.withWriteLock {
|
||||
kvCache.clear()
|
||||
}
|
||||
}
|
||||
|
||||
fun <Key, Value> KeyValuesRepo<Key, Value>.cached(
|
||||
|
@ -9,6 +9,7 @@ import dev.inmo.micro_utils.repos.ReadCRUDRepo
|
||||
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.cache.util.ActualizeAllClearMode
|
||||
import dev.inmo.micro_utils.repos.set
|
||||
import kotlinx.coroutines.CoroutineScope
|
||||
import kotlinx.coroutines.delay
|
||||
@ -56,7 +57,7 @@ open class AutoRecacheReadCRUDRepo<RegisteredObject, Id>(
|
||||
override suspend fun getAll(): Map<Id, RegisteredObject> = actionWrapper.wrap {
|
||||
originalRepo.getAll()
|
||||
}.onSuccess {
|
||||
kvCache.actualizeAll(clear = true) { it }
|
||||
kvCache.actualizeAll(clearMode = ActualizeAllClearMode.BeforeSet) { it }
|
||||
}.getOrElse {
|
||||
kvCache.getAll()
|
||||
}
|
||||
|
@ -9,6 +9,7 @@ import dev.inmo.micro_utils.repos.ReadKeyValueRepo
|
||||
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.cache.util.ActualizeAllClearMode
|
||||
import dev.inmo.micro_utils.repos.set
|
||||
import kotlinx.coroutines.CoroutineScope
|
||||
import kotlinx.coroutines.delay
|
||||
@ -56,7 +57,7 @@ open class AutoRecacheReadKeyValueRepo<Id, RegisteredObject>(
|
||||
override suspend fun getAll(): Map<Id, RegisteredObject> = actionWrapper.wrap {
|
||||
originalRepo.getAll()
|
||||
}.onSuccess {
|
||||
kvCache.actualizeAll(clear = true) { it }
|
||||
kvCache.actualizeAll(clearMode = ActualizeAllClearMode.BeforeSet) { it }
|
||||
}.getOrElse {
|
||||
kvCache.getAll()
|
||||
}
|
||||
|
@ -9,6 +9,7 @@ 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.*
|
||||
import dev.inmo.micro_utils.repos.cache.util.ActualizeAllClearMode
|
||||
import dev.inmo.micro_utils.repos.cache.util.actualizeAll
|
||||
import kotlinx.coroutines.CoroutineScope
|
||||
import kotlinx.coroutines.Dispatchers
|
||||
@ -42,7 +43,7 @@ open class FullReadCRUDCacheRepo<ObjectType, IdType>(
|
||||
)
|
||||
|
||||
protected open suspend fun actualizeAll() {
|
||||
locker.withWriteLock { kvCache.actualizeAll(parentRepo) }
|
||||
kvCache.actualizeAll(parentRepo, locker = locker)
|
||||
}
|
||||
|
||||
override suspend fun getByPagination(pagination: Pagination): PaginationResult<ObjectType> = doOrTakeAndActualize(
|
||||
@ -72,7 +73,7 @@ open class FullReadCRUDCacheRepo<ObjectType, IdType>(
|
||||
override suspend fun getAll(): Map<IdType, ObjectType> = doOrTakeAndActualizeWithWriteLock(
|
||||
{ getAll().takeIf { it.isNotEmpty() }.optionalOrAbsentIfNull },
|
||||
{ getAll() },
|
||||
{ kvCache.actualizeAll(clear = true) { it } }
|
||||
{ kvCache.actualizeAll(clearMode = ActualizeAllClearMode.BeforeSet) { it } }
|
||||
)
|
||||
|
||||
override suspend fun getById(id: IdType): ObjectType? = doOrTakeAndActualizeWithWriteLock(
|
||||
|
@ -8,8 +8,8 @@ import dev.inmo.micro_utils.coroutines.withWriteLock
|
||||
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.util.ActualizeAllClearMode
|
||||
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.flow.*
|
||||
@ -41,10 +41,7 @@ open class FullReadKeyValueCacheRepo<Key,Value>(
|
||||
actualize = { locker.withWriteLock { actualize(it) } }
|
||||
)
|
||||
protected open suspend fun actualizeAll() {
|
||||
locker.withWriteLock {
|
||||
kvCache.clear()
|
||||
kvCache.set(parentRepo.getAll())
|
||||
}
|
||||
kvCache.actualizeAll(parentRepo, locker)
|
||||
}
|
||||
|
||||
override suspend fun get(k: Key): Value? = doOrTakeAndActualizeWithWriteLock(
|
||||
@ -74,7 +71,7 @@ open class FullReadKeyValueCacheRepo<Key,Value>(
|
||||
override suspend fun getAll(): Map<Key, Value> = doOrTakeAndActualizeWithWriteLock(
|
||||
{ getAll().takeIf { it.isNotEmpty() }.optionalOrAbsentIfNull },
|
||||
{ getAll() },
|
||||
{ kvCache.actualizeAll(clear = true) { it } }
|
||||
{ kvCache.actualizeAll(clearMode = ActualizeAllClearMode.BeforeSet) { it } }
|
||||
)
|
||||
|
||||
override suspend fun keys(pagination: Pagination, reversed: Boolean): PaginationResult<Key> = doOrTakeAndActualize(
|
||||
@ -150,9 +147,7 @@ open class FullKeyValueCacheRepo<Key,Value>(
|
||||
override suspend fun unsetWithValues(toUnset: List<Value>) = parentRepo.unsetWithValues(toUnset)
|
||||
|
||||
override suspend fun invalidate() {
|
||||
locker.withWriteLock {
|
||||
kvCache.actualizeAll(parentRepo)
|
||||
}
|
||||
kvCache.actualizeAll(parentRepo, locker)
|
||||
}
|
||||
|
||||
override suspend fun clear() {
|
||||
|
@ -8,6 +8,7 @@ import dev.inmo.micro_utils.coroutines.withWriteLock
|
||||
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.util.ActualizeAllClearMode
|
||||
import dev.inmo.micro_utils.repos.cache.util.actualizeAll
|
||||
import kotlinx.coroutines.CoroutineScope
|
||||
import kotlinx.coroutines.Dispatchers
|
||||
@ -41,15 +42,13 @@ open class FullReadKeyValuesCacheRepo<Key,Value>(
|
||||
)
|
||||
|
||||
protected open suspend fun actualizeKey(k: Key) {
|
||||
locker.withWriteLock {
|
||||
kvCache.set(k, parentRepo.getAll(k))
|
||||
kvCache.actualizeAll(locker = locker, clearMode = ActualizeAllClearMode.Never) {
|
||||
mapOf(k to parentRepo.getAll(k))
|
||||
}
|
||||
}
|
||||
|
||||
protected open suspend fun actualizeAll() {
|
||||
locker.withWriteLock {
|
||||
kvCache.actualizeAll(parentRepo)
|
||||
}
|
||||
kvCache.actualizeAll(parentRepo, locker = locker)
|
||||
}
|
||||
|
||||
override suspend fun get(k: Key, pagination: Pagination, reversed: Boolean): PaginationResult<Value> {
|
||||
@ -187,9 +186,7 @@ open class FullKeyValuesCacheRepo<Key,Value>(
|
||||
}
|
||||
|
||||
override suspend fun invalidate() {
|
||||
locker.withWriteLock {
|
||||
kvCache.actualizeAll(parentRepo)
|
||||
}
|
||||
kvCache.actualizeAll(parentRepo, locker = locker)
|
||||
}
|
||||
|
||||
override suspend fun set(toSet: Map<Key, List<Value>>) {
|
||||
|
@ -1,43 +1,169 @@
|
||||
package dev.inmo.micro_utils.repos.cache.util
|
||||
|
||||
import dev.inmo.micro_utils.coroutines.SmartRWLocker
|
||||
import dev.inmo.micro_utils.coroutines.withWriteLock
|
||||
import dev.inmo.micro_utils.repos.*
|
||||
import kotlinx.serialization.Serializable
|
||||
import kotlin.js.JsName
|
||||
import kotlin.jvm.JvmName
|
||||
|
||||
suspend inline fun <K, V> KeyValueRepo<K, V>.actualizeAll(
|
||||
clear: Boolean = true,
|
||||
/**
|
||||
* `invalidate`/`actualizeAll` clearing mode. Declare when data in original repo will be cleared
|
||||
*/
|
||||
@Serializable
|
||||
sealed interface ActualizeAllClearMode {
|
||||
/**
|
||||
* Instruct user of this mode to clear internal data __before load__ of external data
|
||||
*/
|
||||
@Serializable
|
||||
data object BeforeLoad : ActualizeAllClearMode
|
||||
/**
|
||||
* Instruct user of this mode to clear internal data __after load__ of external data and __before set__ of internal data
|
||||
*/
|
||||
@Serializable
|
||||
data object BeforeSet : ActualizeAllClearMode
|
||||
/**
|
||||
* Instruct user of this mode to never clear internal data
|
||||
*/
|
||||
@Serializable
|
||||
data object Never : ActualizeAllClearMode
|
||||
}
|
||||
|
||||
suspend inline fun <K, V> KeyValueRepo<K, V>.actualizeAllWithClearBeforeLoad(
|
||||
getAll: () -> Map<K, V>
|
||||
) {
|
||||
set(
|
||||
getAll().also {
|
||||
if (clear) {
|
||||
clear()
|
||||
}
|
||||
}
|
||||
)
|
||||
clear()
|
||||
val newData = getAll()
|
||||
set(newData)
|
||||
}
|
||||
|
||||
suspend inline fun <K, V> KeyValueRepo<K, V>.actualizeAllWithClearBeforeSet(
|
||||
getAll: () -> Map<K, V>
|
||||
) {
|
||||
val newData = getAll()
|
||||
clear()
|
||||
set(newData)
|
||||
}
|
||||
|
||||
suspend inline fun <K, V> KeyValueRepo<K, V>.actualizeAllWithoutClear(
|
||||
getAll: () -> Map<K, V>
|
||||
) {
|
||||
val newData = getAll()
|
||||
set(newData)
|
||||
}
|
||||
|
||||
@JvmName("actualizeAllWithClearBeforeLoadWithLocker")
|
||||
@JsName("actualizeAllWithClearBeforeLoadWithLocker")
|
||||
suspend inline fun <K, V> KeyValueRepo<K, V>.actualizeAllWithClearBeforeLoad(
|
||||
locker: SmartRWLocker,
|
||||
getAll: () -> Map<K, V>
|
||||
) {
|
||||
locker.withWriteLock {
|
||||
clear()
|
||||
}
|
||||
|
||||
val newData = getAll()
|
||||
locker.withWriteLock {
|
||||
set(newData)
|
||||
}
|
||||
}
|
||||
|
||||
@JvmName("actualizeAllWithClearBeforeSetWithLocker")
|
||||
@JsName("actualizeAllWithClearBeforeSetWithLocker")
|
||||
suspend inline fun <K, V> KeyValueRepo<K, V>.actualizeAllWithClearBeforeSet(
|
||||
locker: SmartRWLocker,
|
||||
getAll: () -> Map<K, V>
|
||||
) {
|
||||
val newData = getAll()
|
||||
locker.withWriteLock {
|
||||
clear()
|
||||
set(newData)
|
||||
}
|
||||
}
|
||||
|
||||
@JvmName("actualizeAllWithoutClearWithLocker")
|
||||
@JsName("actualizeAllWithoutClearWithLocker")
|
||||
suspend inline fun <K, V> KeyValueRepo<K, V>.actualizeAllWithoutClear(
|
||||
locker: SmartRWLocker,
|
||||
getAll: () -> Map<K, V>
|
||||
) {
|
||||
val newData = getAll()
|
||||
locker.withWriteLock {
|
||||
set(newData)
|
||||
}
|
||||
}
|
||||
|
||||
suspend inline fun <K, V> KeyValueRepo<K, V>.actualizeAllWithClearBeforeLoad(
|
||||
locker: SmartRWLocker? = null,
|
||||
getAll: () -> Map<K, V>
|
||||
) {
|
||||
locker ?.let {
|
||||
actualizeAllWithClearBeforeLoad(locker = locker, getAll)
|
||||
} ?: actualizeAllWithClearBeforeLoad(getAll)
|
||||
}
|
||||
|
||||
suspend inline fun <K, V> KeyValueRepo<K, V>.actualizeAllWithClearBeforeSet(
|
||||
locker: SmartRWLocker? = null,
|
||||
getAll: () -> Map<K, V>
|
||||
) {
|
||||
locker ?.let {
|
||||
actualizeAllWithClearBeforeSet(locker = locker, getAll)
|
||||
} ?: actualizeAllWithClearBeforeSet(getAll)
|
||||
}
|
||||
|
||||
suspend inline fun <K, V> KeyValueRepo<K, V>.actualizeAllWithoutClear(
|
||||
locker: SmartRWLocker? = null,
|
||||
getAll: () -> Map<K, V>
|
||||
) {
|
||||
locker ?.let {
|
||||
actualizeAllWithoutClear(locker = locker, getAll)
|
||||
} ?: actualizeAllWithoutClear(getAll)
|
||||
}
|
||||
|
||||
suspend inline fun <K, V> KeyValueRepo<K, V>.actualizeAll(
|
||||
repo: ReadKeyValueRepo<K, V>,
|
||||
clear: Boolean = true,
|
||||
locker: SmartRWLocker? = null,
|
||||
clearMode: ActualizeAllClearMode = ActualizeAllClearMode.BeforeSet,
|
||||
getAll: () -> Map<K, V>
|
||||
) {
|
||||
actualizeAll(clear) {
|
||||
repo.getAll()
|
||||
when (clearMode) {
|
||||
ActualizeAllClearMode.BeforeLoad -> locker ?.let {
|
||||
actualizeAllWithClearBeforeLoad(locker = locker, getAll)
|
||||
} ?: actualizeAllWithClearBeforeLoad(getAll)
|
||||
ActualizeAllClearMode.BeforeSet -> locker ?.let {
|
||||
actualizeAllWithClearBeforeSet(locker = locker, getAll)
|
||||
} ?: actualizeAllWithClearBeforeSet(getAll)
|
||||
ActualizeAllClearMode.Never -> locker ?.let {
|
||||
actualizeAllWithoutClear(locker = locker, getAll)
|
||||
} ?: actualizeAllWithoutClear(getAll)
|
||||
}
|
||||
}
|
||||
|
||||
suspend inline fun <K, V> KeyValueRepo<K, V>.actualizeAll(
|
||||
parentRepo: ReadKeyValueRepo<K, V>,
|
||||
locker: SmartRWLocker? = null,
|
||||
clearMode: ActualizeAllClearMode = ActualizeAllClearMode.BeforeSet,
|
||||
) {
|
||||
actualizeAll(locker, clearMode) {
|
||||
parentRepo.getAll()
|
||||
}
|
||||
}
|
||||
|
||||
suspend inline fun <K, V> KeyValueRepo<K, List<V>>.actualizeAll(
|
||||
repo: ReadKeyValuesRepo<K, V>,
|
||||
clear: Boolean = true,
|
||||
parentRepo: ReadKeyValuesRepo<K, V>,
|
||||
locker: SmartRWLocker? = null,
|
||||
clearMode: ActualizeAllClearMode = ActualizeAllClearMode.BeforeSet,
|
||||
) {
|
||||
actualizeAll(clear) {
|
||||
repo.getAll()
|
||||
actualizeAll(locker, clearMode) {
|
||||
parentRepo.getAll()
|
||||
}
|
||||
}
|
||||
|
||||
suspend inline fun <K, V> KeyValueRepo<K, V>.actualizeAll(
|
||||
repo: ReadCRUDRepo<V, K>,
|
||||
clear: Boolean = true,
|
||||
parentRepo: ReadCRUDRepo<V, K>,
|
||||
locker: SmartRWLocker? = null,
|
||||
clearMode: ActualizeAllClearMode = ActualizeAllClearMode.BeforeSet,
|
||||
) {
|
||||
actualizeAll(clear) {
|
||||
repo.getAll()
|
||||
actualizeAll(locker, clearMode) {
|
||||
parentRepo.getAll()
|
||||
}
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user