Отправка ответа zio http из функции обратного вызова
Я пытаюсь поиграть с ZIO http, используя их простой пример hello world. У меня есть написанная на Java служба, которая выполняет некоторую логику и ожидает функции обработчика, поэтому она может вызвать ее, когда результат будет готов. Как использовать его вместе с ZIO http? Я хочу что-то вроде этого:
object HelloWorld extends App {
def app(service: JavaService) = Http.collect[Request] {
case Method.GET -> Root / "text" => {
service.doSomeStuffWIthCallback((s:String) => Response.text(s))
}
}
override def run(args: List[String]): URIO[zio.ZEnv, ExitCode] =
Server.start(8090, app(new JavaService)).exitCode
}
В основном я хочу отправить HTTP-ответ ZIO от функции обратного вызова, я просто не уверен, как это сделать. Спасибо.
3 ответа
Вы должны обернуть свою службу Java обратным вызовом в эффекте, используя
effectAsync
:
def doSomeStuffWrapped(service: JavaService): Task[String] = {
IO.effectAsync[Throwable, String] { cb =>
service.doSomeStuffWithCallback((s: String) => {
// Success case
cb(IO.succeed(s))
// Optional error case?
// cb(IO.fail(someException))
})
}
}
def app(service: JavaService) = Http.collectM[Request] {
case Method.GET -> Root / "text" => {
doSomeStuffWrapped(service)
.fold(err => {
// Handle errors in some way
Response.text("An error occured")
}, successStr => {
Response.text(successStr)
})
}
}
Возможно, вы захотите увидеть эту статью, в которой представлены различные варианты упаковки нечистого кода в ZIO: https://medium.com/@ghostdogpr/wrapping-impure-code-with-zio-9265c219e2e .
В ZIO-http
v1.0.0.0-RC18
HttpData.fromStream также может принимать ZStream[R, E, String] в качестве входных данных с кодировкой Http, которая по умолчанию равна
CharsetUtil.UTF_8
однако вы можете передать любую кодировку в HttpData.fromStream в качестве второго аргумента. вы можете найти решение ниже
val stream: ZStream[Any, Nothing, String] = ZStream.fromEffect(doSomeStuffWrapped)
val content: HttpData[Any, Nothing] = HttpData.fromStream(stream)
def doSomeStuffWrapped = {
UIO.effectAsync[String] { cb =>
cb(
IO.succeed("TEST STRING"),
)
}
}
// Create HTTP route
val app = Http.collect[Request] {
case Method.GET -> !! / "health" => Response.ok
case Method.GET -> !! / "file" => Response(data = content)
}
// Run it like any simple app
override def run(args: List[String]): URIO[zio.ZEnv, ExitCode] =
Server.start(8090, app.silent).exitCode
Однако в предыдущих версиях вы могли сделать что-то вроде приведенного ниже, чтобы заставить его работать.
val stream: ZStream[Any, Nothing, Byte] =
ZStream.fromEffect(doSomeStuffWrapped).mapChunks(_.map(x => Chunk.fromArray(x.getBytes(HTTP_CHARSET))).flatten)
val content: HttpData[Any, Nothing] = HttpData.fromStream(stream)
def doSomeStuffWrapped = {
UIO.effectAsync[String] { cb =>
cb(
IO.succeed("TEST STRING"),
)
}
}
// Create HTTP route
val app = Http.collect[Request] {
case Method.GET -> !! / "health" => Response.ok
case Method.GET -> !! / "file" => Response(data = content)
}
// Run it like any simple app
override def run(args: List[String]): URIO[zio.ZEnv, ExitCode] =
Server.start(8090, app.silent).exitCode
Вот еще один способ достижения того же результата:
case class MyService(name: String) {
def imDone[R, E](s: String => Unit): Unit = s(name)
}
val s: MyService = MyService("test")
val app: Http[Any, Nothing, Request, UResponse] = Http.collectM[Request] { case Method.GET -> Root / "text" =>
ZIO.effectAsync[Any, Nothing, UResponse] { cb =>
s.imDone { b =>
cb(IO.succeed(Response.text(b)))
}
}
}