akka http客户端的断路器示例看起来非常简单,对我来说并不适用。
object HttpWithCircuitBreaker extends App {
implicit val system = ActorSystem()
implicit val materializer = ActorMaterializer()
implicit val ec = system.dispatcher
val breaker =
new CircuitBreaker(
system.scheduler,
maxFailures = 2,
callTimeout = 3.seconds,
resetTimeout = 25.seconds)
.onOpen(println("circuit breaker opened"))
.onClose(println("circuit breaker closed"))
.onHalfOpen(println("circuit breaker half-open"))
while (true) {
val futureResponse: Future[HttpResponse] = Http().singleRequest(HttpRequest(uri = "https://www.random.org/integers/?num=1&min=1&max=6&col=1&base=10&format=plain&rnd=new"))
breaker.withCircuitBreaker(futureResponse).map(resp => resp.status match {
case Success(_) =>
resp.entity.dataBytes.runWith(Sink.ignore)
println("http success")
case _ =>
resp.entity.dataBytes.runWith(Sink.ignore)
println(s"http error ${resp.status.intValue()}")
}).recover {
case e@_ =>
println(s"exception ${e.getMessage}")
}
Thread.sleep(1000)
}
}
它开始非常精细地获取随机数。它在断开网络时打开,但在重新连接时从不关闭。正如您在日志中看到的那样,尝试恢复但是失败并超时:
http success
http success
http success
exception Tcp command [Connect(www.random.org:443,None,List(),Some(10 seconds),true)] failed
exception Circuit Breaker Timed out.
circuit breaker opened
exception Tcp command [Connect(www.random.org:443,None,List(),Some(10 seconds),true)] failed
exception Circuit Breaker is open; calls are failing fast
exception Circuit Breaker Timed out.
exception Circuit Breaker is open; calls are failing fast
exception Circuit Breaker is open; calls are failing fast
exception Circuit Breaker is open; calls are failing fast
exception Circuit Breaker is open; calls are failing fast
exception Circuit Breaker is open; calls are failing fast
exception Circuit Breaker is open; calls are failing fast
exception Circuit Breaker is open; calls are failing fast
exception Circuit Breaker is open; calls are failing fast
exception Circuit Breaker is open; calls are failing fast
exception Circuit Breaker is open; calls are failing fast
exception Circuit Breaker is open; calls are failing fast
exception Circuit Breaker is open; calls are failing fast
exception Circuit Breaker is open; calls are failing fast
exception Circuit Breaker is open; calls are failing fast
exception Circuit Breaker is open; calls are failing fast
exception Circuit Breaker is open; calls are failing fast
exception Circuit Breaker is open; calls are failing fast
exception Circuit Breaker is open; calls are failing fast
exception Circuit Breaker is open; calls are failing fast
exception Circuit Breaker is open; calls are failing fast
exception Circuit Breaker is open; calls are failing fast
exception Circuit Breaker is open; calls are failing fast
exception Circuit Breaker is open; calls are failing fast
exception Circuit Breaker is open; calls are failing fast
circuit breaker half-open <--- new http call should start here
exception Circuit Breaker is open; calls are failing fast
exception Circuit Breaker is open; calls are failing fast
exception Circuit Breaker is open; calls are failing fast
exception Circuit Breaker Timed out. <--- but it fails with timeout
circuit breaker opened
exception Circuit Breaker is open; calls are failing fast
exception Circuit Breaker is open; calls are failing fast
exception Circuit Breaker is open; calls are failing fast
exception Circuit Breaker is open; calls are failing fast
exception Circuit Breaker is open; calls are failing fast
exception Circuit Breaker is open; calls are failing fast
Gut的感觉说它一定是未经消费的http实体,但我仍然无法做到正确。
答案 0 :(得分:3)
要使断路器正常工作,请val
val futureResponse: Future[HttpResponse] = Http().singleRequest(...)
应为def
def futureResponse: Future[HttpResponse] = Http().singleRequest(...)
断路器需要包装异步调用 - 如果你使用val
,你的异步调用将在断路器之外启动。
除此之外:不要在代码中使用Thread.sleep,尝试使用 - 例如 - 阿卡after。