我仍然是使用Play!Framework 2.1 WS库实现Web服务请求的新手。现在,我在理解WS库行为方面存在问题。
首先,我有这样的代码:
public static Result espnRss() {
try {
// do request
return async(
WS.url("http://espnfc.com/rss/news?section=premierleague").get().map(
new F.Function<WS.Response, Result>() {
@Override
public Result apply(WS.Response response) throws Throwable {
return ok("Success!"); // success request
}
}
)
);
} catch (Exception e) {
// exception occured
return internalServerError("Oops, connect exception occured!");
}
}
当我尝试请求espnRss
操作时,我收到了SUCCESS响应。
然后,我想在请求上设置WS超时。所以,我改变了我之前的代码:
public static Result espnRss() {
try {
// set request timeout for 1000 ms and do request
return async(
WS.url("http://espnfc.com/rss/news?section=premierleague").setTimeout(1000).get().map(
... // same as previous
)
);
} catch (Exception e) {
// exception occured
return internalServerError("Oops, connect exception occured!");
}
}
我的互联网连接速度不快(下载速度约为40 KB / s),我故意这样做(设置请求超时1秒),以便执行异常处理代码。
但是,我从框架获得默认响应,而不是internalServerError
响应提供的代码。
Execution Exception
[TimeoutException: No response received after 1000]
任何人都可以解释为什么使用上面的代码无法捕获WS请求上的异常吗?使用Play!Framework WS库处理异常的最佳方法是什么?
答案 0 :(得分:9)
要处理异步请求(例如使用Play!Framework 2.1.0的WS
请求)发生的异常,Promise
上的方法名为recover(F.Function<java.lang.Throwable,A> function)
。
当我们想要处理请求使用WS
库时发生的所有异常时,应该调用该方法。所以,我使用如下代码解决了问题:
public static Result espnRss() {
// do request
return async(
WS.url("http://espnfc.com/rss/news?section=premierleague").setTimeout(100).get().map(
new F.Function<WS.Response, Result>() {
@Override
public Result apply(WS.Response response) throws Throwable {
return ok("Success!"); // success request
}
}
).recover( // to handle error occured on redeemed PROMISE
new F.Function<Throwable, Result>() {
@Override
public Result apply(Throwable throwable) throws Throwable {
// option to distinguish exception
if (throwable instanceof TimeoutException) {
return internalServerError("Oops, time out exception occured!");
} else {
return internalServerError("Oops, other exception occured!");
}
}
}
)
);
}
答案 1 :(得分:2)
我不熟悉Play框架,但async
必须返回/使用某种未来。该请求实际上是在一个单独的线程中执行的,显然没有try..catch
处理程序捕获到异常。
必须有一些像onComplete
这样的函数/方法可以应用于async
,允许您测试运行请求的结果。