telegramium.bots.high.BotApi.scala Maven / Gradle / Ivy
package telegramium.bots.high
import cats.effect.Async
import cats.effect.Sync
import cats.syntax.all.*
import io.circe.Decoder
import io.circe.Json
import iozhik.DecodingError
import org.http4s.Request
import org.http4s.Uri
import org.http4s.circe.*
import org.http4s.client.*
import org.http4s.dsl.io.*
import org.http4s.multipart.Part
import telegramium.bots.InputPartFile
import telegramium.bots.client.Method
import telegramium.bots.high.Http4sUtils.toFileDataParts
import telegramium.bots.high.Http4sUtils.toMultipartWithFormData
class BotApi[F[_]](
http: Client[F],
baseUrl: String
)(implicit F: Async[F])
extends Api[F] {
override def execute[Res](method: Method[Res]): F[Res] = {
val inputPartFiles = method.payload.files.collect { case (filename, InputPartFile(file)) =>
(filename, file)
}
val attachments = toFileDataParts(inputPartFiles)
for {
uri <- F.fromEither[Uri](Uri.fromString(s"$baseUrl/${method.payload.name}"))
req <- mkRequest(uri, method.payload.json, inputPartFiles.keys.toList, attachments)
res <- handleResponse[Res](method, req)(using method.decoder)
} yield res
}
private def mkRequest(uri: Uri, json: Json, fileFieldNames: List[String], attachments: Vector[Part[F]]) = {
val reqBuilder = Request[F]().withMethod(POST).withUri(uri)
if (attachments.isEmpty) reqBuilder.withEntity(json).pure[F]
else
toMultipartWithFormData(json, fileFieldNames, attachments).map { parts =>
reqBuilder.withEntity(parts).withHeaders(parts.headers)
}
}
private case class Response[A](
ok: Boolean,
result: Option[A],
description: Option[String],
errorCode: Option[Int]
)
private implicit def responseDecoder[A: Decoder]: Decoder[Response[A]] =
Decoder.instance { h =>
for {
_ok <- h.get[Boolean]("ok")
_result <- h.get[Option[A]]("result")
_description <- h.get[Option[String]]("description")
_errorCode <- h.get[Option[Int]]("error_code")
} yield {
Response[A](ok = _ok, result = _result, description = _description, errorCode = _errorCode)
}
}
private def handleResponse[A: io.circe.Decoder](method: Method[A], req: Request[F]): F[A] =
for {
response <- http
.fetchAs(req)(jsonOf[F, Response[A]])
.adaptError { case e @ DecodingError(message) =>
ResponseDecodingError.default(message, e.some)
}
result <- response match {
case Response(true, Some(result), _, _) => F.pure(result)
case Response(_, _, description, errorCode) =>
val code = errorCode.map(_.toString).getOrElse("")
val desc = description.getOrElse("")
val methodName = method.payload.name
val json = method.payload.json
FailedRequest(method, errorCode, description).raiseError[F, A]
}
} yield result
}
object BotApi {
def apply[F[_]: Async](http: Client[F], baseUrl: String): BotApi[F] =
new BotApi[F](http, baseUrl)
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy