88 lines
3.7 KiB
Kotlin
88 lines
3.7 KiB
Kotlin
package com.github.insanusmokrassar.TelegramBotAPI.bot.Ktor
|
|
|
|
import com.github.insanusmokrassar.TelegramBotAPI.bot.BaseRequestsExecutor
|
|
import com.github.insanusmokrassar.TelegramBotAPI.bot.Ktor.base.MultipartRequestCallFactory
|
|
import com.github.insanusmokrassar.TelegramBotAPI.bot.Ktor.base.SimpleRequestCallFactory
|
|
import com.github.insanusmokrassar.TelegramBotAPI.bot.exceptions.newRequestException
|
|
import com.github.insanusmokrassar.TelegramBotAPI.bot.settings.limiters.EmptyLimiter
|
|
import com.github.insanusmokrassar.TelegramBotAPI.bot.settings.limiters.RequestLimiter
|
|
import com.github.insanusmokrassar.TelegramBotAPI.requests.abstracts.Request
|
|
import com.github.insanusmokrassar.TelegramBotAPI.types.Response
|
|
import com.github.insanusmokrassar.TelegramBotAPI.types.RetryAfterError
|
|
import com.github.insanusmokrassar.TelegramBotAPI.utils.TelegramAPIUrlsKeeper
|
|
import io.ktor.client.HttpClient
|
|
import io.ktor.client.call.receive
|
|
import io.ktor.client.features.ClientRequestException
|
|
import io.ktor.client.statement.HttpStatement
|
|
import io.ktor.client.statement.readText
|
|
import kotlinx.coroutines.delay
|
|
import kotlinx.serialization.json.Json
|
|
|
|
class KtorRequestsExecutor(
|
|
telegramAPIUrlsKeeper: TelegramAPIUrlsKeeper,
|
|
private val client: HttpClient = HttpClient(),
|
|
callsFactories: List<KtorCallFactory> = emptyList(),
|
|
excludeDefaultFactories: Boolean = false,
|
|
private val requestsLimiter: RequestLimiter = EmptyLimiter,
|
|
private val jsonFormatter: Json = Json.nonstrict
|
|
) : BaseRequestsExecutor(telegramAPIUrlsKeeper) {
|
|
private val callsFactories: List<KtorCallFactory> = callsFactories.run {
|
|
if (!excludeDefaultFactories) {
|
|
asSequence().plus(SimpleRequestCallFactory()).plus(MultipartRequestCallFactory()).toList()
|
|
} else {
|
|
this
|
|
}
|
|
}
|
|
|
|
override suspend fun <T : Any> execute(request: Request<T>): T {
|
|
return requestsLimiter.limit {
|
|
var statement: HttpStatement? = null
|
|
for (factory in callsFactories) {
|
|
statement = factory.prepareCall(
|
|
client,
|
|
telegramAPIUrlsKeeper.commonAPIUrl,
|
|
request
|
|
)
|
|
if (statement != null) {
|
|
break
|
|
}
|
|
}
|
|
try {
|
|
val response = statement ?.execute() ?: throw IllegalArgumentException("Can't execute request: $request")
|
|
val content = response.receive<String>()
|
|
val responseObject = jsonFormatter.parse(Response.serializer(), content)
|
|
|
|
(responseObject.result?.let {
|
|
jsonFormatter.fromJson(request.resultDeserializer, it)
|
|
} ?: responseObject.parameters?.let {
|
|
val error = it.error
|
|
if (error is RetryAfterError) {
|
|
delay(error.leftToRetry)
|
|
execute(request)
|
|
} else {
|
|
null
|
|
}
|
|
} ?: response.let {
|
|
throw newRequestException(
|
|
responseObject,
|
|
content,
|
|
"Can't get result object from $content"
|
|
)
|
|
})
|
|
} catch (e: ClientRequestException) {
|
|
val content = e.response.readText()
|
|
val responseObject = jsonFormatter.parse(Response.serializer(), content)
|
|
throw newRequestException(
|
|
responseObject,
|
|
content,
|
|
"Can't get result object from $content"
|
|
)
|
|
}
|
|
}
|
|
}
|
|
|
|
override fun close() {
|
|
client.close()
|
|
}
|
|
}
|