forked from github/wulkanowy-mirror
Refactor networkBoundResource (#2482)
--------- Co-authored-by: Faierbel <RafalBO99@outlook.com>
This commit is contained in:
parent
6a8f6f9496
commit
8a90b61b97
@ -9,6 +9,7 @@ import kotlinx.coroutines.flow.combine
|
|||||||
import kotlinx.coroutines.flow.debounce
|
import kotlinx.coroutines.flow.debounce
|
||||||
import kotlinx.coroutines.flow.emitAll
|
import kotlinx.coroutines.flow.emitAll
|
||||||
import kotlinx.coroutines.flow.filter
|
import kotlinx.coroutines.flow.filter
|
||||||
|
import kotlinx.coroutines.flow.filterNot
|
||||||
import kotlinx.coroutines.flow.first
|
import kotlinx.coroutines.flow.first
|
||||||
import kotlinx.coroutines.flow.flatMapLatest
|
import kotlinx.coroutines.flow.flatMapLatest
|
||||||
import kotlinx.coroutines.flow.flow
|
import kotlinx.coroutines.flow.flow
|
||||||
@ -22,15 +23,15 @@ import timber.log.Timber
|
|||||||
import kotlin.time.Duration
|
import kotlin.time.Duration
|
||||||
import kotlin.time.Duration.Companion.seconds
|
import kotlin.time.Duration.Companion.seconds
|
||||||
|
|
||||||
sealed class Resource<out T> {
|
sealed interface Resource<out T> {
|
||||||
|
|
||||||
open class Loading<T> : Resource<T>()
|
open class Loading<T> : Resource<T>
|
||||||
|
|
||||||
data class Intermediate<T>(val data: T) : Loading<T>()
|
data class Intermediate<T>(val data: T) : Loading<T>()
|
||||||
|
|
||||||
data class Success<T>(val data: T) : Resource<T>()
|
data class Success<T>(val data: T) : Resource<T>
|
||||||
|
|
||||||
data class Error<T>(val error: Throwable) : Resource<T>()
|
data class Error<T>(val error: Throwable) : Resource<T>
|
||||||
}
|
}
|
||||||
|
|
||||||
val <T> Resource<T>.dataOrNull: T?
|
val <T> Resource<T>.dataOrNull: T?
|
||||||
@ -97,7 +98,7 @@ fun <T> Flow<Resource<T>>.logResourceStatus(name: String, showData: Boolean = fa
|
|||||||
Timber.i("$name: $description")
|
Timber.i("$name: $description")
|
||||||
}
|
}
|
||||||
|
|
||||||
fun <T, U> Flow<Resource<T>>.mapResourceData(block: suspend (T) -> U) = map {
|
inline fun <T, U> Flow<Resource<T>>.mapResourceData(crossinline block: suspend (T) -> U) = map {
|
||||||
when (it) {
|
when (it) {
|
||||||
is Resource.Success -> Resource.Success(block(it.data))
|
is Resource.Success -> Resource.Success(block(it.data))
|
||||||
is Resource.Intermediate -> Resource.Intermediate(block(it.data))
|
is Resource.Intermediate -> Resource.Intermediate(block(it.data))
|
||||||
@ -167,33 +168,32 @@ suspend fun <T> Flow<Resource<T>>.waitForResult() = takeWhile { it is Resource.L
|
|||||||
|
|
||||||
// Can cause excessive amounts of `Resource.Intermediate` to be emitted. Unless that is desired,
|
// Can cause excessive amounts of `Resource.Intermediate` to be emitted. Unless that is desired,
|
||||||
// use `debounceIntermediates` to alleviate this behavior.
|
// use `debounceIntermediates` to alleviate this behavior.
|
||||||
inline fun <reified T> combineResourceFlows(
|
inline fun <reified T> combineResourceFlows(flows: Iterable<Flow<Resource<T>>>): Flow<Resource<List<T>>> =
|
||||||
flows: Iterable<Flow<Resource<T>>>,
|
combine(flows) { items ->
|
||||||
): Flow<Resource<List<T>>> = combine(flows) { items ->
|
var isIntermediate = false
|
||||||
var isIntermediate = false
|
val data = mutableListOf<T>()
|
||||||
val data = mutableListOf<T>()
|
for (item in items) {
|
||||||
for (item in items) {
|
when (item) {
|
||||||
when (item) {
|
is Resource.Success -> data.add(item.data)
|
||||||
is Resource.Success -> data.add(item.data)
|
is Resource.Intermediate -> {
|
||||||
is Resource.Intermediate -> {
|
isIntermediate = true
|
||||||
isIntermediate = true
|
data.add(item.data)
|
||||||
data.add(item.data)
|
}
|
||||||
}
|
|
||||||
|
|
||||||
is Resource.Loading -> return@combine Resource.Loading()
|
is Resource.Loading -> return@combine Resource.Loading()
|
||||||
is Resource.Error -> continue
|
is Resource.Error -> continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (data.isEmpty()) {
|
||||||
|
// All items have to be errors for this to happen, so just return the first one.
|
||||||
|
// mapData is functionally useless and exists only to satisfy the type checker
|
||||||
|
items.first().mapData { listOf(it) }
|
||||||
|
} else if (isIntermediate) {
|
||||||
|
Resource.Intermediate(data)
|
||||||
|
} else {
|
||||||
|
Resource.Success(data)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (data.isEmpty()) {
|
|
||||||
// All items have to be errors for this to happen, so just return the first one.
|
|
||||||
// mapData is functionally useless and exists only to satisfy the type checker
|
|
||||||
items.first().mapData { listOf(it) }
|
|
||||||
} else if (isIntermediate) {
|
|
||||||
Resource.Intermediate(data)
|
|
||||||
} else {
|
|
||||||
Resource.Success(data)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@OptIn(FlowPreview::class)
|
@OptIn(FlowPreview::class)
|
||||||
fun <T> Flow<Resource<T>>.debounceIntermediates(timeout: Duration = 5.seconds) = flow {
|
fun <T> Flow<Resource<T>>.debounceIntermediates(timeout: Duration = 5.seconds) = flow {
|
||||||
@ -214,70 +214,51 @@ fun <T> Flow<Resource<T>>.debounceIntermediates(timeout: Duration = 5.seconds) =
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
inline fun <ResultType, RequestType> networkBoundResource(
|
inline fun <ResultType, RequestType> networkBoundResource(
|
||||||
mutex: Mutex = Mutex(),
|
mutex: Mutex = Mutex(),
|
||||||
showSavedOnLoading: Boolean = true,
|
|
||||||
crossinline isResultEmpty: (ResultType) -> Boolean,
|
crossinline isResultEmpty: (ResultType) -> Boolean,
|
||||||
crossinline query: () -> Flow<ResultType>,
|
crossinline query: () -> Flow<ResultType>,
|
||||||
crossinline fetch: suspend (ResultType) -> RequestType,
|
crossinline fetch: suspend () -> RequestType,
|
||||||
crossinline saveFetchResult: suspend (old: ResultType, new: RequestType) -> Unit,
|
crossinline saveFetchResult: suspend (old: ResultType, new: RequestType) -> Unit,
|
||||||
crossinline onFetchFailed: (Throwable) -> Unit = { },
|
|
||||||
crossinline shouldFetch: (ResultType) -> Boolean = { true },
|
crossinline shouldFetch: (ResultType) -> Boolean = { true },
|
||||||
crossinline filterResult: (ResultType) -> ResultType = { it }
|
crossinline filterResult: (ResultType) -> ResultType = { it }
|
||||||
) = flow {
|
) = networkBoundResource(
|
||||||
emit(Resource.Loading())
|
mutex = mutex,
|
||||||
|
isResultEmpty = isResultEmpty,
|
||||||
val data = query().first()
|
query = query,
|
||||||
emitAll(if (shouldFetch(data)) {
|
fetch = fetch,
|
||||||
val filteredResult = filterResult(data)
|
saveFetchResult = saveFetchResult,
|
||||||
|
shouldFetch = shouldFetch,
|
||||||
if (showSavedOnLoading && !isResultEmpty(filteredResult)) {
|
mapResult = filterResult
|
||||||
emit(Resource.Intermediate(filteredResult))
|
)
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
val newData = fetch(data)
|
|
||||||
mutex.withLock { saveFetchResult(query().first(), newData) }
|
|
||||||
query().map { Resource.Success(filterResult(it)) }
|
|
||||||
} catch (throwable: Throwable) {
|
|
||||||
onFetchFailed(throwable)
|
|
||||||
flowOf(Resource.Error(throwable))
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
query().map { Resource.Success(filterResult(it)) }
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
@JvmName("networkBoundResourceWithMap")
|
@JvmName("networkBoundResourceWithMap")
|
||||||
inline fun <ResultType, RequestType, T> networkBoundResource(
|
inline fun <ResultType, RequestType, MappedResultType> networkBoundResource(
|
||||||
mutex: Mutex = Mutex(),
|
mutex: Mutex = Mutex(),
|
||||||
showSavedOnLoading: Boolean = true,
|
crossinline isResultEmpty: (MappedResultType) -> Boolean,
|
||||||
crossinline isResultEmpty: (T) -> Boolean,
|
|
||||||
crossinline query: () -> Flow<ResultType>,
|
crossinline query: () -> Flow<ResultType>,
|
||||||
crossinline fetch: suspend (ResultType) -> RequestType,
|
crossinline fetch: suspend () -> RequestType,
|
||||||
crossinline saveFetchResult: suspend (old: ResultType, new: RequestType) -> Unit,
|
crossinline saveFetchResult: suspend (old: ResultType, new: RequestType) -> Unit,
|
||||||
crossinline onFetchFailed: (Throwable) -> Unit = { },
|
|
||||||
crossinline shouldFetch: (ResultType) -> Boolean = { true },
|
crossinline shouldFetch: (ResultType) -> Boolean = { true },
|
||||||
crossinline mapResult: (ResultType) -> T,
|
crossinline mapResult: (ResultType) -> MappedResultType,
|
||||||
) = flow {
|
) = flow {
|
||||||
emit(Resource.Loading())
|
emit(Resource.Loading())
|
||||||
|
|
||||||
val data = query().first()
|
val data = query().first()
|
||||||
emitAll(if (shouldFetch(data)) {
|
if (shouldFetch(data)) {
|
||||||
val mappedResult = mapResult(data)
|
emit(Resource.Intermediate(data))
|
||||||
|
|
||||||
if (showSavedOnLoading && !isResultEmpty(mappedResult)) {
|
|
||||||
emit(Resource.Intermediate(mappedResult))
|
|
||||||
}
|
|
||||||
try {
|
try {
|
||||||
val newData = fetch(data)
|
val newData = fetch()
|
||||||
mutex.withLock { saveFetchResult(query().first(), newData) }
|
mutex.withLock { saveFetchResult(query().first(), newData) }
|
||||||
query().map { Resource.Success(mapResult(it)) }
|
|
||||||
} catch (throwable: Throwable) {
|
} catch (throwable: Throwable) {
|
||||||
onFetchFailed(throwable)
|
emit(Resource.Error(throwable))
|
||||||
flowOf(Resource.Error(throwable))
|
return@flow
|
||||||
}
|
}
|
||||||
} else {
|
}
|
||||||
query().map { Resource.Success(mapResult(it)) }
|
|
||||||
})
|
emitAll(query().map { Resource.Success(it) })
|
||||||
}
|
}
|
||||||
|
.mapResourceData { mapResult(it) }
|
||||||
|
.filterNot { it is Resource.Intermediate && isResultEmpty(it.data) }
|
||||||
|
@ -6,6 +6,7 @@ import io.github.wulkanowy.data.db.dao.AdminMessageDao
|
|||||||
import io.github.wulkanowy.data.db.entities.AdminMessage
|
import io.github.wulkanowy.data.db.entities.AdminMessage
|
||||||
import io.github.wulkanowy.data.networkBoundResource
|
import io.github.wulkanowy.data.networkBoundResource
|
||||||
import kotlinx.coroutines.flow.Flow
|
import kotlinx.coroutines.flow.Flow
|
||||||
|
import kotlinx.coroutines.flow.filterNot
|
||||||
import kotlinx.coroutines.sync.Mutex
|
import kotlinx.coroutines.sync.Mutex
|
||||||
import javax.inject.Inject
|
import javax.inject.Inject
|
||||||
import javax.inject.Singleton
|
import javax.inject.Singleton
|
||||||
@ -28,6 +29,6 @@ class AdminMessageRepository @Inject constructor(
|
|||||||
saveFetchResult = { oldItems, newItems ->
|
saveFetchResult = { oldItems, newItems ->
|
||||||
adminMessageDao.removeOldAndSaveNew(oldItems, newItems)
|
adminMessageDao.removeOldAndSaveNew(oldItems, newItems)
|
||||||
},
|
},
|
||||||
showSavedOnLoading = false,
|
|
||||||
)
|
)
|
||||||
|
.filterNot { it is Resource.Intermediate }
|
||||||
}
|
}
|
||||||
|
@ -122,7 +122,7 @@ class MessageRepository @Inject constructor(
|
|||||||
fetch = {
|
fetch = {
|
||||||
wulkanowySdkFactory.create(student)
|
wulkanowySdkFactory.create(student)
|
||||||
.getMessageDetails(
|
.getMessageDetails(
|
||||||
messageKey = it!!.message.messageGlobalKey,
|
messageKey = message.messageGlobalKey,
|
||||||
markAsRead = message.unread && markAsRead,
|
markAsRead = message.unread && markAsRead,
|
||||||
)
|
)
|
||||||
},
|
},
|
||||||
|
@ -1,6 +1,10 @@
|
|||||||
package io.github.wulkanowy.data
|
package io.github.wulkanowy.data
|
||||||
|
|
||||||
import io.mockk.*
|
import io.mockk.Runs
|
||||||
|
import io.mockk.coEvery
|
||||||
|
import io.mockk.coVerifyOrder
|
||||||
|
import io.mockk.just
|
||||||
|
import io.mockk.mockk
|
||||||
import kotlinx.coroutines.ExperimentalCoroutinesApi
|
import kotlinx.coroutines.ExperimentalCoroutinesApi
|
||||||
import kotlinx.coroutines.delay
|
import kotlinx.coroutines.delay
|
||||||
import kotlinx.coroutines.flow.flowOf
|
import kotlinx.coroutines.flow.flowOf
|
||||||
@ -42,7 +46,6 @@ class ResourceTest {
|
|||||||
// first
|
// first
|
||||||
networkBoundResource(
|
networkBoundResource(
|
||||||
isResultEmpty = { false },
|
isResultEmpty = { false },
|
||||||
showSavedOnLoading = false,
|
|
||||||
query = { repo.query() },
|
query = { repo.query() },
|
||||||
fetch = {
|
fetch = {
|
||||||
val data = repo.fetch()
|
val data = repo.fetch()
|
||||||
@ -57,7 +60,6 @@ class ResourceTest {
|
|||||||
// second
|
// second
|
||||||
networkBoundResource(
|
networkBoundResource(
|
||||||
isResultEmpty = { false },
|
isResultEmpty = { false },
|
||||||
showSavedOnLoading = false,
|
|
||||||
query = { repo.query() },
|
query = { repo.query() },
|
||||||
fetch = {
|
fetch = {
|
||||||
val data = repo.fetch()
|
val data = repo.fetch()
|
||||||
@ -124,7 +126,6 @@ class ResourceTest {
|
|||||||
networkBoundResource(
|
networkBoundResource(
|
||||||
isResultEmpty = { false },
|
isResultEmpty = { false },
|
||||||
mutex = saveResultMutex,
|
mutex = saveResultMutex,
|
||||||
showSavedOnLoading = false,
|
|
||||||
query = { repo.query() },
|
query = { repo.query() },
|
||||||
fetch = {
|
fetch = {
|
||||||
val data = repo.fetch()
|
val data = repo.fetch()
|
||||||
@ -143,7 +144,6 @@ class ResourceTest {
|
|||||||
networkBoundResource(
|
networkBoundResource(
|
||||||
isResultEmpty = { false },
|
isResultEmpty = { false },
|
||||||
mutex = saveResultMutex,
|
mutex = saveResultMutex,
|
||||||
showSavedOnLoading = false,
|
|
||||||
query = { repo.query() },
|
query = { repo.query() },
|
||||||
fetch = {
|
fetch = {
|
||||||
val data = repo.fetch()
|
val data = repo.fetch()
|
||||||
|
Loading…
x
Reference in New Issue
Block a user