【问题标题】:How to listen websocket server close events using akka-http websocket client如何使用 akka-http websocket 客户端监听 websocket 服务器关闭事件
【发布时间】:2016-06-09 13:29:34
【问题描述】:

我有 websocket 客户端连接到 akka-http websocket 服务器,我如何监听服务器上发生的连接关闭事件(即服务器关闭/服务器关闭 websocket 连接)?

object Client extends App {


  implicit val actorSystem = ActorSystem("akka-system")
  implicit val flowMaterializer = ActorMaterializer()

  val config = actorSystem.settings.config
  val interface = config.getString("app.interface")

  val port = config.getInt("app.port")


  // print each incoming strict text message
  val printSink: Sink[Message, Future[Done]] =
    Sink.foreach {
      case message: TextMessage.Strict =>
        println(message.text)

      case _ => {
        sourceQueue.map(q => {
          println(s"offering message on client")
          q.offer(TextMessage("received unknown"))
        })
        println(s"received unknown message format")
      }
    }

  val (source, sourceQueue) = {
    val p = Promise[SourceQueue[Message]]
    val s = Source.queue[Message](Int.MaxValue, OverflowStrategy.backpressure).mapMaterializedValue(m => {
      p.trySuccess(m)
      m
    })
      .keepAlive(FiniteDuration(1, TimeUnit.SECONDS), () => TextMessage.Strict("Heart Beat"))
    (s, p.future)
  }

  val flow =
    Flow.fromSinkAndSourceMat(printSink, source)(Keep.right)


  val (upgradeResponse, sourceClose) =
    Http().singleWebSocketRequest(WebSocketRequest("ws://localhost:8080/ws-echo"), flow)

  val connected = upgradeResponse.map { upgrade =>
    // just like a regular http request we can get 404 NotFound,
    // with a response body, that will be available from upgrade.response
    if (upgrade.response.status == StatusCodes.SwitchingProtocols || upgrade.response.status == StatusCodes.SwitchingProtocols) {
      Done
    } else {
      throw new RuntimeException(s"Connection failed: ${upgrade.response.status}")
    }
  }


  connected.onComplete(println)

}

【问题讨论】:

    标签: scala websocket akka-http


    【解决方案1】:

    Websocket 连接终止被建模为常规流完成,因此在您的情况下,您可以使用由Sink.foreach 产生的具体化Future[Done]

    val flow = Flow.fromSinkAndSourceMat(printSink, source)(Keep.both)
    
    val (upgradeResponse, (sinkClose, sourceClose)) =
      Http().singleWebSocketRequest(..., flow)
    
    sinkClose.onComplete {
      case Success(_) => println("Connection closed gracefully")
      case Failure(e) => println("Connection closed with an error: $e")
    }
    

    【讨论】:

    • 这取决于您的架构。我认为最简单的方法是将singleWebSocketRequest() 调用和sinkClose.onComplete 处理程序提取到一个方法中,并从onComplete 处理程序中递归调用此方法。
    • 好的,我希望他们支持开箱即用,如 socket.io *.com/questions/13797262/…
    • 您的回答准确无误。谢谢!