Отправка ответа 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-RC18HttpData.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)))
      }
    }
  }
Другие вопросы по тегам