Как правильно отправить ответ JSON в http4s? - PullRequest
3 голосов
/ 23 мая 2019

Не так давно я перешел с akka-http на http4s. Одна из основных вещей, которую я хотел сделать правильно - обработка JSON, в частности отправка ответа JSON.

Я решил использовать http4s с ZIO вместо кошек, поэтому вот как выглядит http-маршрут:

import fs2.Stream
import org.http4s._
import org.http4s.dsl.io._
import org.http4s.implicits._
import scalaz.zio.Task
import scalaz.zio.interop.catz._
import io.circe.generic.auto._
import io.circe.syntax._

class TweetsRoutes {

  case class Tweet(author: String, tweet: String)

  val helloWorldService = HttpRoutes.of[Task] {
    case GET -> Root / "hello" / name => Task {
      Response[Task](Ok)
        .withBodyStream(Stream.emits(
          Tweet(name, "dummy tweet text").asJson.toString.getBytes
        ))
    }
  }.orNotFound

}

Как видите, часть сериализации JSON довольно многословна:

.withBodyStream(Stream.emits(
  Tweet(name, "dummy tweet text").asJson.toString.getBytes
))

Есть ли другой способ отправить JSON в ответе?

Ответы [ 2 ]

5 голосов
/ 23 мая 2019

Да, есть: define и Encoder and Decoder for Task:

implicit def circeJsonDecoder[A](
      implicit decoder: Decoder[A]
  ): EntityDecoder[Task, A] = jsonOf[Task, A]
  implicit def circeJsonEncoder[A](
      implicit encoder: Encoder[A]
  ): EntityEncoder[Task, A] = jsonEncoderOf[Task, A]

, поэтому нет необходимости преобразовывать в байты.

EDIT: здесь приведен полный пример: https://github.com/mschuwalow/zio-todo-backend/blob/develop/src/main/scala/com/schuwalow/zio/todo/http/TodoService.scala

HT: @ mschuwalow

0 голосов
/ 12 июня 2019

Для этого есть еще более простое решение.Если вы хотите обработать кодировку JSON класса дел для ответов HTTP, вы можете просто добавить эти импорты:

import io.circe.generic.auto._
import org.http4s.circe.CirceEntityCodec._

Кстати, эти же импорты обрабатывают декодирование входящих запросов JSON и в классы дел

...