All Downloads are FREE. Search and download functionalities are using the official Maven repository.

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