Skip to content

Commit

Permalink
Encode SSE based on HttpContentCodec (zio#2695)
Browse files Browse the repository at this point in the history
  • Loading branch information
987Nabil committed Jul 13, 2024
1 parent e692d78 commit 9439ec1
Show file tree
Hide file tree
Showing 2 changed files with 29 additions and 28 deletions.
23 changes: 22 additions & 1 deletion zio-http/shared/src/main/scala/zio/http/ServerSentEvent.scala
Original file line number Diff line number Diff line change
Expand Up @@ -16,10 +16,15 @@

package zio.http

import zio.stacktracer.TracingImplicits.disableAutoTrace
import zio._

import zio.stream.ZPipeline

import zio.schema.codec.{BinaryCodec, DecodeError}
import zio.schema.{DeriveSchema, Schema}

import zio.http.codec.{BinaryCodecWithSchema, HttpContentCodec}

/**
* Server-Sent Event (SSE) as defined by
* https://html.spec.whatwg.org/multipage/server-sent-events.html#server-sent-events
Expand Down Expand Up @@ -61,5 +66,21 @@ final case class ServerSentEvent(
object ServerSentEvent {
implicit lazy val schema: Schema[ServerSentEvent] = DeriveSchema.gen[ServerSentEvent]

implicit val contentCodec: HttpContentCodec[ServerSentEvent] = HttpContentCodec.from(
MediaType.text.`event-stream` -> BinaryCodecWithSchema.fromBinaryCodec(new BinaryCodec[ServerSentEvent] {
override def decode(whole: Chunk[Byte]): Either[DecodeError, ServerSentEvent] =
throw new UnsupportedOperationException("ServerSentEvent decoding is not yet supported.")

override def streamDecoder: ZPipeline[Any, DecodeError, Byte, ServerSentEvent] =
throw new UnsupportedOperationException("ServerSentEvent decoding is not yet supported.")

override def encode(value: ServerSentEvent): Chunk[Byte] =
Chunk.fromArray(value.encode.getBytes)

override def streamEncoder: ZPipeline[Any, Nothing, ServerSentEvent, Byte] =
ZPipeline.mapChunks(value => value.flatMap(c => c.encode.getBytes))
}),
)

def heartbeat: ServerSentEvent = new ServerSentEvent("")
}
Original file line number Diff line number Diff line change
Expand Up @@ -216,11 +216,6 @@ private[codec] object EncoderDecoder {
} else {
false
}
private val isEventStream = if (flattened.content.length == 1) {
isEventStreamBody(flattened.content(0))
} else {
false
}
private val onlyTheLastFieldIsStreaming =
if (flattened.content.size > 1) {
!flattened.content.init.exists(isByteStreamBody) && isByteStreamBody(flattened.content.last)
Expand Down Expand Up @@ -533,26 +528,20 @@ private[codec] object EncoderDecoder {
case SimpleCodec.Specified(method) => Some(method)
}
} else None
private def encodeBody(inputs: Array[Any], outputTypes: Chunk[MediaTypeWithQFactor]): Body = {
private def encodeBody(inputs: Array[Any], outputTypes: Chunk[MediaTypeWithQFactor]): Body =
if (isByteStream) {
Body.fromStreamChunked(inputs(0).asInstanceOf[ZStream[Any, Nothing, Byte]])
} else {
if (inputs.length > 1) {
Body.fromMultipartForm(encodeMultipartFormData(inputs, outputTypes), formBoundary)
} else {
if (isEventStream) {
Body.fromCharSequenceStreamChunked(
inputs(0).asInstanceOf[ZStream[Any, Nothing, ServerSentEvent]].map(_.encode),
)
} else if (inputs.length < 1) {
inputs.length match {
case 0 =>
Body.empty
} else {
case 1 =>
val bodyCodec = flattened.content(0)
bodyCodec.erase.encodeToBody(inputs(0), outputTypes)
}
case _ =>
Body.fromMultipartForm(encodeMultipartFormData(inputs, outputTypes), formBoundary)
}
}
}

private def encodeMultipartFormData(inputs: Array[Any], outputTypes: Chunk[MediaTypeWithQFactor]): Form = {
Form(
Expand Down Expand Up @@ -581,8 +570,7 @@ private[codec] object EncoderDecoder {
if (inputs.length > 1) {
Headers(Header.ContentType(MediaType.multipart.`form-data`))
} else {
if (isEventStream) Headers(Header.ContentType(MediaType.text.`event-stream`))
else if (flattened.content.length < 1) Headers.empty
if (flattened.content.length < 1) Headers.empty
else {
val mediaType = flattened
.content(0)
Expand All @@ -599,14 +587,6 @@ private[codec] object EncoderDecoder {
case BodyCodec.Multiple(codec, _) if codec.defaultMediaType.binary => true
case _ => false
}

private def isEventStreamBody(codec: BodyCodec[_]): Boolean =
codec match {
case BodyCodec.Multiple(codec, _)
if codec.lookup(MediaType.text.`event-stream`).exists(_.schema == Schema[ServerSentEvent]) =>
true
case _ => false
}
}

}

0 comments on commit 9439ec1

Please sign in to comment.