org.http4s.play.PlayInstances.scala Maven / Gradle / Ivy
package org.http4s.play
import cats.Applicative
import cats.effect.Sync
import fs2.Chunk
import org.http4s.headers.`Content-Type`
import org.http4s.{
DecodeResult,
EntityDecoder,
EntityEncoder,
InvalidMessageBodyFailure,
MediaType,
Message,
Uri,
jawn
}
import org.typelevel.jawn.support.play.Parser.facade
import play.api.libs.json._
trait PlayInstances {
def jsonOf[F[_]: Sync, A](implicit decoder: Reads[A]): EntityDecoder[F, A] =
jsonDecoder[F].flatMapR { json =>
decoder
.reads(json)
.fold(
_ =>
DecodeResult.failure(InvalidMessageBodyFailure(s"Could not decode JSON: $json", None)),
DecodeResult.success(_)
)
}
implicit def jsonDecoder[F[_]: Sync]: EntityDecoder[F, JsValue] =
jawn.jawnDecoder[F, JsValue]
def jsonEncoderOf[F[_]: EntityEncoder[?[_], String]: Applicative, A: Writes]
: EntityEncoder[F, A] =
jsonEncoder[F].contramap[A](Json.toJson(_))
implicit def jsonEncoder[F[_]: Applicative]: EntityEncoder[F, JsValue] =
EntityEncoder[F, Chunk[Byte]]
.contramap[JsValue] { json =>
val bytes = json.toString.getBytes("UTF8")
Chunk.bytes(bytes)
}
.withContentType(`Content-Type`(MediaType.application.json))
implicit val writesUri: Writes[Uri] =
Writes.contravariantfunctorWrites.contramap[String, Uri](implicitly[Writes[String]], _.toString)
implicit val readsUri: Reads[Uri] =
implicitly[Reads[String]].flatMap { str =>
Uri
.fromString(str)
.fold(
_ =>
new Reads[Uri] {
def reads(json: JsValue): JsResult[Uri] = JsError("Invalid uri")
},
Reads.pure(_)
)
}
implicit class MessageSyntax[F[_]: Sync](self: Message[F]) {
def decodeJson[A: Reads]: F[A] =
self.as(implicitly, jsonOf[F, A])
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy