5

在我们的项目中,我们使用ReactiveMongowith Play 2.2.1

问题是,以 的形式Enumerator[A]返回的数据ReactiveMongo流实际上是一个值对象流,没有逗号分隔,也没有流的开始和结束注解,可以看作数组的开闭陈述。

JSON这给消费者带来了问题JS client,因为预期的格式是 [A1,A2, ...]

所以我们跳了起来,将我们的Enumeratee[A]to转换为Enumerator[String],并检查它是否是第一个元素:

    var first:Boolean = true
    val aToStrs = (as.map(a => {
        if(first) {
            first = false;
            Json.stringify(Json.toJson(a))
        } else {
            "," + Json.stringify(Json.toJson(a))
        }
   }))
   Ok.chunked(
       Enumerator.enumInput(Input.El("[")) andThen
       aToStrs andThen
       Enumerator.enumInput(Input.El("]")) andThen
       Enumerator.enumInput(Input.EOF)
   )

这行得通,但感觉就像在发明轮子。

对于这个常见问题,是否有更好的解决方案?

4

1 回答 1

1

如果您使用 comet 或 EventSource,您将不必手工制作一种生成输出的方法,并且您实际上还可以解析客户端中项目的响应项目。使用数组响应将迫使您编写自己的解析代码或等到所有内容都到达客户端,然后才能在 JavaScript 中使用 build int JSON 解析器。

使用 EventSource 协议进行流式传输非常容易,您应该能够执行以下操作:

implicit val cometEncoder = new Comet.CometMessage[JsValue](_.toString)
Ok.chunked(yourEnumerator &> EventSource()).as(EVENT_STREAM)

然后在客户端html中:

<script type="text/javascript">
  var es = new EventSource(jsRouter.controllers.Application.streamIt().url)

  es.addEventListener("message", function (event) {
    var item = JSON.parse(event.data)
    // ... do something with the json value ...
  })
</script>

您可能还想查看播放示例项目中的一个示例$YOUR_PLAY_DIR/samples/scala/eventsource-clock/

于 2014-02-26T15:02:46.870 回答