Magrate schedule domain to flow
This commit is contained in:
@@ -6,10 +6,13 @@ import androidx.activity.ComponentActivity
|
||||
import androidx.lifecycle.lifecycleScope
|
||||
import dagger.hilt.android.AndroidEntryPoint
|
||||
import kotlinx.coroutines.launch
|
||||
import ru.fincode.tsudesk.core.common.model.DataResult
|
||||
import ru.fincode.tsudesk.feature.schedule.domain.model.ScheduleType
|
||||
import ru.fincode.tsudesk.feature.schedule.domain.usecase.GetScheduleUseCase
|
||||
import javax.inject.Inject
|
||||
|
||||
private const val LOG_TAG = "NETWORK_DEBUG"
|
||||
|
||||
@AndroidEntryPoint
|
||||
class MainActivity : ComponentActivity() {
|
||||
@Inject
|
||||
@@ -19,13 +22,22 @@ class MainActivity : ComponentActivity() {
|
||||
super.onCreate(savedInstanceState)
|
||||
|
||||
lifecycleScope.launch {
|
||||
try {
|
||||
val result = getScheduleUseCase(
|
||||
ScheduleType.Group("220631")
|
||||
)
|
||||
Log.d("TSUDesk", result.toString())
|
||||
} catch (e: Exception) {
|
||||
Log.e("TSUDesk", "Error loading schedule", e)
|
||||
getScheduleUseCase(
|
||||
ScheduleType.Group("220631") // пример группы
|
||||
).collect { result ->
|
||||
when (result) {
|
||||
is DataResult.Cache -> {
|
||||
Log.d(LOG_TAG, "CACHE: ${result.data}")
|
||||
}
|
||||
|
||||
is DataResult.Network -> {
|
||||
Log.d(LOG_TAG, "NETWORK: ${result.data}")
|
||||
}
|
||||
|
||||
is DataResult.Error -> {
|
||||
Log.e(LOG_TAG, "ERROR: ${result.throwable}")
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -5,7 +5,7 @@ import dagger.Provides
|
||||
import dagger.hilt.InstallIn
|
||||
import dagger.hilt.components.SingletonComponent
|
||||
import ru.fincode.tsudesk.BuildConfig
|
||||
import ru.fincode.tsudesk.core.common.config.AppConfig
|
||||
import ru.fincode.tsudesk.core.common.app.AppConfig
|
||||
import javax.inject.Singleton
|
||||
|
||||
@Module
|
||||
|
||||
@@ -1,4 +1,4 @@
|
||||
package ru.fincode.tsudesk.core.common.config
|
||||
package ru.fincode.tsudesk.core.common.app
|
||||
|
||||
data class AppConfig(
|
||||
val isDebug: Boolean,
|
||||
@@ -0,0 +1,16 @@
|
||||
package ru.fincode.tsudesk.core.common.model
|
||||
|
||||
sealed interface DataResult<out T> {
|
||||
|
||||
data class Cache<T>(
|
||||
val data: T
|
||||
) : DataResult<T>
|
||||
|
||||
data class Network<T>(
|
||||
val data: T
|
||||
) : DataResult<T>
|
||||
|
||||
data class Error(
|
||||
val throwable: Throwable
|
||||
) : DataResult<Nothing>
|
||||
}
|
||||
@@ -1,49 +1,55 @@
|
||||
package ru.fincode.tsudesk.core.database.schedule
|
||||
|
||||
import androidx.room.Dao
|
||||
import androidx.room.Delete
|
||||
import androidx.room.Insert
|
||||
import androidx.room.OnConflictStrategy
|
||||
import androidx.room.Query
|
||||
import androidx.room.Transaction
|
||||
import ru.fincode.tsudesk.core.database.schedule.Query.DELETE_LESSON_BY_GROUP_KEY_QUERY
|
||||
import kotlinx.coroutines.flow.Flow
|
||||
import ru.fincode.tsudesk.core.database.schedule.Query.SELECT_LESSON_BY_KEY_QUERY
|
||||
import ru.fincode.tsudesk.core.database.schedule.Query.SELECT_SCHEDULE_BY_KEY_QUERY
|
||||
|
||||
@Dao
|
||||
interface ScheduleDao {
|
||||
|
||||
@Insert(onConflict = OnConflictStrategy.Companion.REPLACE)
|
||||
suspend fun upsertSchedule(schedule: ScheduleCacheEntity)
|
||||
|
||||
@Insert(onConflict = OnConflictStrategy.Companion.REPLACE)
|
||||
suspend fun insertLessons(lessons: List<LessonCacheEntity>)
|
||||
|
||||
@Query(SELECT_SCHEDULE_BY_KEY_QUERY)
|
||||
suspend fun getSchedule(key: String): ScheduleCacheEntity?
|
||||
fun observeSchedule(key: String): Flow<ScheduleCacheEntity?>
|
||||
|
||||
@Query(SELECT_LESSON_BY_KEY_QUERY)
|
||||
fun observeLessons(key: String): Flow<List<LessonCacheEntity>>
|
||||
|
||||
@Query("SELECT * FROM schedule_cache WHERE `key` = :key LIMIT 1")
|
||||
suspend fun getSchedule(key: String): ScheduleCacheEntity?
|
||||
|
||||
@Query("SELECT * FROM lesson_cache WHERE scheduleKey = :key")
|
||||
suspend fun getLessons(key: String): List<LessonCacheEntity>
|
||||
|
||||
@Query(DELETE_LESSON_BY_GROUP_KEY_QUERY)
|
||||
@Insert(onConflict = OnConflictStrategy.REPLACE)
|
||||
suspend fun upsertSchedule(schedule: ScheduleCacheEntity)
|
||||
|
||||
@Insert(onConflict = OnConflictStrategy.REPLACE)
|
||||
suspend fun insertLessons(lessons: List<LessonCacheEntity>)
|
||||
|
||||
@Query("DELETE FROM lesson_cache WHERE scheduleKey = :key")
|
||||
suspend fun deleteLessonsByKey(key: String)
|
||||
|
||||
@Delete
|
||||
suspend fun deleteSchedule(entity: ScheduleCacheEntity)
|
||||
@Query("DELETE FROM schedule_cache WHERE `key` = :key")
|
||||
suspend fun deleteScheduleByKey(key: String)
|
||||
|
||||
@Transaction
|
||||
suspend fun replaceSchedule(
|
||||
key: String, schedule: ScheduleCacheEntity, lessons: List<LessonCacheEntity>
|
||||
key: String,
|
||||
schedule: ScheduleCacheEntity,
|
||||
lessons: List<LessonCacheEntity>
|
||||
) {
|
||||
upsertSchedule(schedule)
|
||||
deleteLessonsByKey(key)
|
||||
upsertSchedule(schedule)
|
||||
insertLessons(lessons)
|
||||
}
|
||||
|
||||
@Transaction
|
||||
suspend fun clearSchedule(key: String) {
|
||||
deleteLessonsByKey(key)
|
||||
val header = getSchedule(key) ?: return
|
||||
deleteSchedule(header)
|
||||
deleteScheduleByKey(key)
|
||||
}
|
||||
}
|
||||
@@ -1,7 +1,7 @@
|
||||
package ru.fincode.tsudesk.core.network
|
||||
|
||||
import okhttp3.OkHttpClient
|
||||
import ru.fincode.tsudesk.core.common.config.AppConfig
|
||||
import ru.fincode.tsudesk.core.common.app.AppConfig
|
||||
import ru.fincode.tsudesk.core.network.interceptor.DebugInterceptor
|
||||
import java.util.concurrent.TimeUnit
|
||||
import javax.inject.Inject
|
||||
|
||||
@@ -6,7 +6,7 @@ import dagger.hilt.InstallIn
|
||||
import dagger.hilt.components.SingletonComponent
|
||||
import okhttp3.OkHttpClient
|
||||
import retrofit2.Retrofit
|
||||
import ru.fincode.tsudesk.core.common.config.AppConfig
|
||||
import ru.fincode.tsudesk.core.common.app.AppConfig
|
||||
import ru.fincode.tsudesk.core.network.RetrofitFactory
|
||||
import javax.inject.Singleton
|
||||
|
||||
|
||||
@@ -43,4 +43,5 @@ dependencies {
|
||||
|
||||
implementation(project(":core:network"))
|
||||
implementation(project(":core:database"))
|
||||
implementation(project(":core:common"))
|
||||
}
|
||||
@@ -1,5 +1,13 @@
|
||||
package ru.fincode.tsudesk.feature.schedule.data
|
||||
|
||||
import android.os.Build
|
||||
import androidx.annotation.RequiresApi
|
||||
import kotlinx.coroutines.channels.awaitClose
|
||||
import kotlinx.coroutines.flow.Flow
|
||||
import kotlinx.coroutines.flow.channelFlow
|
||||
import kotlinx.coroutines.launch
|
||||
import ru.fincode.tsudesk.core.common.model.DataResult
|
||||
import ru.fincode.tsudesk.core.common.model.DataResult.Error
|
||||
import ru.fincode.tsudesk.core.network.model.NetworkResult
|
||||
import ru.fincode.tsudesk.core.network.model.map
|
||||
import ru.fincode.tsudesk.feature.schedule.data.datasource.ScheduleLocalDataSource
|
||||
@@ -7,6 +15,7 @@ import ru.fincode.tsudesk.feature.schedule.data.datasource.ScheduleRemoteDataSou
|
||||
import ru.fincode.tsudesk.feature.schedule.data.mapper.ScheduleCacheKey
|
||||
import ru.fincode.tsudesk.feature.schedule.data.mapper.ScheduleNetworkMapper
|
||||
import ru.fincode.tsudesk.feature.schedule.domain.model.ScheduleEntity
|
||||
import ru.fincode.tsudesk.feature.schedule.domain.model.ScheduleType
|
||||
import ru.fincode.tsudesk.feature.schedule.domain.repository.ScheduleRepository
|
||||
import javax.inject.Inject
|
||||
|
||||
@@ -16,19 +25,50 @@ class ScheduleRepositoryImpl @Inject constructor(
|
||||
private val mapper: ScheduleNetworkMapper
|
||||
) : ScheduleRepository {
|
||||
|
||||
override suspend fun loadScheduleByGroup(number: String): NetworkResult<ScheduleEntity> {
|
||||
val result = remote.loadScheduleByGroup(number).map(mapper::invoke)
|
||||
if (result is NetworkResult.Success) {
|
||||
local.saveSchedule(ScheduleCacheKey.group(number), result.data)
|
||||
}
|
||||
return result
|
||||
@RequiresApi(Build.VERSION_CODES.O)
|
||||
override fun observeSchedule(type: ScheduleType): Flow<DataResult<ScheduleEntity>> =
|
||||
channelFlow {
|
||||
val key = when (type) {
|
||||
is ScheduleType.Group -> ScheduleCacheKey.group(type.number)
|
||||
is ScheduleType.Teacher -> ScheduleCacheKey.teacher(type.name)
|
||||
}
|
||||
|
||||
override suspend fun loadScheduleByTeacher(name: String): NetworkResult<ScheduleEntity> {
|
||||
val result = remote.loadScheduleByTeacher(name).map(mapper::invoke)
|
||||
if (result is NetworkResult.Success) {
|
||||
local.saveSchedule(ScheduleCacheKey.teacher(name), result.data)
|
||||
}
|
||||
return result
|
||||
// кэш -> UI
|
||||
val cacheJob = launch {
|
||||
local.observeSchedule(key).collect { cached ->
|
||||
if (cached != null) send(DataResult.Cache(cached))
|
||||
}
|
||||
}
|
||||
|
||||
val networkResult: NetworkResult<ScheduleEntity> = when (type) {
|
||||
is ScheduleType.Group ->
|
||||
remote.loadScheduleByGroup(type.number).map(mapper::invoke)
|
||||
|
||||
is ScheduleType.Teacher ->
|
||||
remote.loadScheduleByTeacher(type.name).map(mapper::invoke)
|
||||
}
|
||||
|
||||
when (networkResult) {
|
||||
is NetworkResult.Success -> {
|
||||
// (опционально) сразу отдать сетевой результат
|
||||
send(DataResult.Network(networkResult.data))
|
||||
|
||||
// single source of truth -> сохраняем в БД
|
||||
local.saveSchedule(key, networkResult.data)
|
||||
|
||||
// дать Room время эмитнуть обновление
|
||||
kotlinx.coroutines.yield()
|
||||
|
||||
// закрываем поток по твоему требованию
|
||||
close()
|
||||
}
|
||||
|
||||
is NetworkResult.Error -> {
|
||||
send(Error(Throwable(networkResult.error.toString())))
|
||||
close()
|
||||
}
|
||||
}
|
||||
|
||||
awaitClose { cacheJob.cancel() }
|
||||
}
|
||||
}
|
||||
@@ -1,9 +1,11 @@
|
||||
package ru.fincode.tsudesk.feature.schedule.data.datasource
|
||||
|
||||
import kotlinx.coroutines.flow.Flow
|
||||
import ru.fincode.tsudesk.feature.schedule.domain.model.ScheduleEntity
|
||||
|
||||
interface ScheduleLocalDataSource {
|
||||
suspend fun saveSchedule(key: String, schedule: ScheduleEntity)
|
||||
suspend fun loadSchedule(key: String): ScheduleEntity?
|
||||
fun observeSchedule(key: String): Flow<ScheduleEntity?>
|
||||
suspend fun removeSchedule(key: String)
|
||||
}
|
||||
|
||||
@@ -1,5 +1,7 @@
|
||||
package ru.fincode.tsudesk.feature.schedule.data.local
|
||||
|
||||
import kotlinx.coroutines.flow.Flow
|
||||
import kotlinx.coroutines.flow.combine
|
||||
import ru.fincode.tsudesk.core.database.schedule.ScheduleDao
|
||||
import ru.fincode.tsudesk.feature.schedule.data.datasource.ScheduleLocalDataSource
|
||||
import ru.fincode.tsudesk.feature.schedule.data.mapper.toCache
|
||||
@@ -22,6 +24,13 @@ class ScheduleLocalDataSourceImpl @Inject constructor(
|
||||
return schedule.toDomain(lessons)
|
||||
}
|
||||
|
||||
override fun observeSchedule(key: String): Flow<ScheduleEntity?> {
|
||||
return dao.observeSchedule(key)
|
||||
.combine(dao.observeLessons(key)) { schedule, lessons ->
|
||||
schedule?.toDomain(lessons)
|
||||
}
|
||||
}
|
||||
|
||||
override suspend fun removeSchedule(key: String) {
|
||||
dao.deleteLessonsByKey(key)
|
||||
}
|
||||
|
||||
@@ -1,9 +1,10 @@
|
||||
package ru.fincode.tsudesk.feature.schedule.domain.repository
|
||||
|
||||
import ru.fincode.tsudesk.core.network.model.NetworkResult
|
||||
import kotlinx.coroutines.flow.Flow
|
||||
import ru.fincode.tsudesk.core.common.model.DataResult
|
||||
import ru.fincode.tsudesk.feature.schedule.domain.model.ScheduleEntity
|
||||
import ru.fincode.tsudesk.feature.schedule.domain.model.ScheduleType
|
||||
|
||||
interface ScheduleRepository {
|
||||
suspend fun loadScheduleByGroup(number: String): NetworkResult<ScheduleEntity>
|
||||
suspend fun loadScheduleByTeacher(name: String): NetworkResult<ScheduleEntity>
|
||||
fun observeSchedule(type: ScheduleType): Flow<DataResult<ScheduleEntity>>
|
||||
}
|
||||
|
||||
@@ -1,6 +1,7 @@
|
||||
package ru.fincode.tsudesk.feature.schedule.domain.usecase
|
||||
|
||||
import ru.fincode.tsudesk.core.network.model.NetworkResult
|
||||
import kotlinx.coroutines.flow.Flow
|
||||
import ru.fincode.tsudesk.core.common.model.DataResult
|
||||
import ru.fincode.tsudesk.feature.schedule.domain.model.ScheduleEntity
|
||||
import ru.fincode.tsudesk.feature.schedule.domain.model.ScheduleType
|
||||
import ru.fincode.tsudesk.feature.schedule.domain.repository.ScheduleRepository
|
||||
@@ -9,8 +10,6 @@ import javax.inject.Inject
|
||||
class GetScheduleUseCase @Inject constructor(
|
||||
private val repository: ScheduleRepository
|
||||
) {
|
||||
suspend operator fun invoke(type: ScheduleType): NetworkResult<ScheduleEntity> = when (type) {
|
||||
is ScheduleType.Group -> repository.loadScheduleByGroup(type.number)
|
||||
is ScheduleType.Teacher -> repository.loadScheduleByTeacher(type.name)
|
||||
}
|
||||
operator fun invoke(type: ScheduleType): Flow<DataResult<ScheduleEntity>> =
|
||||
repository.observeSchedule(type)
|
||||
}
|
||||
|
||||
@@ -24,17 +24,13 @@ lifecycle = "2.7.0"
|
||||
coroutines = "1.8.1"
|
||||
|
||||
room = "2.6.1"
|
||||
junit = "4.13.2"
|
||||
junitVersion = "1.1.5"
|
||||
espressoCore = "3.5.1"
|
||||
|
||||
|
||||
[libraries]
|
||||
# Android
|
||||
androidx-lifecycle-runtime-ktx = { group = "androidx.lifecycle", name = "lifecycle-runtime-ktx", version.ref = "lifecycle" }
|
||||
kotlinx-coroutines-android = { group = "org.jetbrains.kotlinx", name = "kotlinx-coroutines-android", version.ref = "coroutines" }
|
||||
#UI: AndroidX, Jetpack Compose
|
||||
androidx-appcompat = { group = "androidx.appcompat", name = "appcompat", version.ref = "appcompat" }
|
||||
# UI: AndroidX, Jetpack Compose
|
||||
material = { group = "com.google.android.material", name = "material", version.ref = "material" }
|
||||
androidx-activity = { group = "androidx.activity", name = "activity", version.ref = "activity" }
|
||||
androidx-constraintlayout = { group = "androidx.constraintlayout", name = "constraintlayout", version.ref = "constraintlayout" }
|
||||
|
||||
Reference in New Issue
Block a user