package com.github.apognu.otter.repositories import android.net.Uri import com.github.apognu.otter.utils.* import com.github.kittinunf.fuel.Fuel import com.github.kittinunf.fuel.core.FuelError import com.github.kittinunf.fuel.core.ResponseDeserializable import com.github.kittinunf.fuel.coroutines.awaitObjectResponseResult import com.github.kittinunf.fuel.coroutines.awaitObjectResult import com.github.kittinunf.result.Result import com.google.gson.Gson import com.preference.PowerPreference import kotlinx.coroutines.Dispatchers.IO import kotlinx.coroutines.flow.Flow import kotlinx.coroutines.flow.flow import kotlinx.coroutines.flow.flowOn import java.io.Reader import java.lang.reflect.Type import kotlin.math.ceil class HttpUpstream>(val behavior: Behavior, private val url: String, private val type: Type) : Upstream { enum class Behavior { Single, AtOnce, Progressive } override fun fetch(size: Int): Flow> = flow { if (behavior == Behavior.Single && size != 0) return@flow val page = ceil(size / AppContext.PAGE_SIZE.toDouble()).toInt() + 1 val offsetUrl = Uri.parse(url) .buildUpon() .appendQueryParameter("page_size", AppContext.PAGE_SIZE.toString()) .appendQueryParameter("page", page.toString()) .build() .toString() get(offsetUrl).fold( { response -> val data = response.getData() if (behavior == Behavior.Progressive || response.next == null) { emit(Repository.Response(Repository.Origin.Network, data, false)) } else { emit(Repository.Response(Repository.Origin.Network, data, true)) fetch(size + data.size) } }, { error -> when (error.exception) { is RefreshError -> EventBus.send(Event.LogOut) else -> emit(Repository.Response(Repository.Origin.Network, listOf(), false)) } } ) }.flowOn(IO) class GenericDeserializer>(val type: Type) : ResponseDeserializable { override fun deserialize(reader: Reader): T? { return Gson().fromJson(reader, type) } } suspend fun get(url: String): Result { val token = PowerPreference.getFileByName(AppContext.PREFS_CREDENTIALS).getString("access_token") val (_, response, result) = Fuel .get(mustNormalizeUrl(url)) .header("Authorization", "Bearer $token") .awaitObjectResponseResult(GenericDeserializer(type)) if (response.statusCode == 401) { return retryGet(url) } return result } private suspend fun retryGet(url: String): Result { return if (HTTP.refresh()) { val token = PowerPreference.getFileByName(AppContext.PREFS_CREDENTIALS).getString("access_token") Fuel .get(mustNormalizeUrl(url)) .header("Authorization", "Bearer $token") .awaitObjectResult(GenericDeserializer(type)) } else { Result.Failure(FuelError.wrap(RefreshError)) } } }