2

我正在尝试将使用 Play 版本 2.4 的代码转换为当前版本(2.6),但我遇到了一些问题,因为我仍然是 Scala 中的菜鸟。

def wsWeatherIntervals = WebSocket.using[String] {
  request =>
    val url = "http://api.openweathermap.org/data/2.5/weather?q=Amsterdam,nl"
    val outEnumerator = Enumerator.repeatM[String]({
      Thread.sleep(3000)
      ws.url(url).get().map(r => s"${new java.util.Date()}\n ${r.body}")
    })
    (Iteratee.ignore[String], outEnumerator)
}

我遵循了这个指南,但现在我被困在我应该在方法上返回的东西上。这是我尝试使用 2.6 版运行的代码:

  import play.api.mvc._
  import scala.concurrent.Future
  import akka.stream.scaladsl._
  def wsWeatherIntervals = WebSocket.accept[String, Future[String]] { res =>
    val url = "http://api.openweathermap.org/data/2.5/weather?q=Amsterdam,nl"
    val source = Source.repeat({
      Thread.sleep(3000)
      ws.url(url).get().map(r => s"${new java.util.Date()}\n ${r.body}")
    })
    Flow.fromSinkAndSource(Sink.ignore, source)
  }

但是我在运行服务器时遇到了这个错误,它指向方法的第一行:

could not find implicit value for parameter transformer: play.api.mvc.WebSocket.MessageFlowTransformer[String,scala.concurrent.Future[String]]

注意:我也尝试调用WebSocket.apply而不是,WebSocket.accept我对两者之间的差异进行了一些搜索,但没有发现任何有用的东西。有人可以解释两者之间的区别吗?谢谢。

4

1 回答 1

2

表面上的错误是 Play 不知道如何将 aFuture[String]转换为 Websocket 消息,您通常会为此使用隐式转换器。但是,在这种情况下,您无论如何都不想返回 a Future[String],而只想返回一个可以自动编组的普通字符串(使用提供的stringMessageFlowTransformer,因为它发生了。)下面是应该工作的东西:

def wsWeatherIntervals = WebSocket.accept[String, String] { res =>
  val url = "http://api.openweathermap.org/data/2.5/weather?q=Amsterdam,nl"
  def f = ws.url(url).get().map(r => s"${new java.util.Date()}\n ${r.body}")

  val source = Source.unfoldAsync(f)(last => {
    Thread.sleep(3000)
    f.map(next => Some((last, next)))
  })
  Flow.fromSinkAndSource(Sink.ignore, source)
}

源代码让我们重复运行一个函数,unfoldAsync返回流中下一个元素的未来。(因为我们希望流永远持续下去,所以我们返回包装为 的值Some。)

Websocket.apply方法基本上是一个更复杂的版本,accept它允许您出于某种原因通过返回响应来拒绝 websocket 连接,但如果您需要这样做,最好使用acceptOrResult,它处理将您的流发出的任何内容转换为 websocket 消息。

于 2017-09-28T15:25:06.513 回答