语言:java
UI驱动:selenium/appium
问题现象:批量执行Chrome的UI自动化,长时间后,程序会卡住;
解决该问题,首先先了解selenium驱动浏览器操作的底层原理是什么,以查找某个页面元素为例,以下为代码:
一层一层跟进代码发现,是调用的一次Http请求,再继续查看这个client实现,发现是基于Okhttp3的,而在这里就分别定义了连接与读的超时时间为2分钟与3个小时,而这个3个小时等待 也就是程序会卡住的原因,所以缩短这个超时时间使用例尽快失败可能会是一种好的方式,而后续就要看看怎么把这个超时时间设置成我们的;
解决问题:
查看webdrvier的初始化过程,如下:
好像找到入口了,继续查看这个HttpCommandExecutor如何创建一个我们所需要的,如下:
看到这里,基本就明朗了,只需要我们自定义一个Factory,传入这个构造函数中,就可以自定义我们所需要的executor了,查看Factory的实现与我们上边的截图一致,所以我们照样画瓢就可以了,以下为实现代码:
实现自己的Factory,定义好需要的超时时间:
public class HttpClientFactory implements org.openqa.selenium.remote.http.HttpClient.Factory{
private final ConnectionPool pool = new ConnectionPool();
private final Duration connectionTimeout = Duration.ofMinutes(2L);
private final Duration readTimeout = Duration.ofMinutes(5L);
@Override
public HttpClient.Builder builder() {
return new org.openqa.selenium.remote.http.HttpClient.Builder() {
public HttpClient createClient(URL url) {
okhttp3.OkHttpClient.Builder client = (new okhttp3.OkHttpClient.Builder()).connectionPool(HttpClientFactory.this.pool).followRedirects(true).followSslRedirects(true).proxy(this.proxy).readTimeout(HttpClientFactory.this.readTimeout.toMillis(), TimeUnit.MILLISECONDS).connectTimeout(HttpClientFactory.this.connectionTimeout.toMillis(), TimeUnit.MILLISECONDS);
String info = url.getUserInfo();
if (!Strings.isNullOrEmpty(info)) {
String[] parts = info.split(":", 2);
String user = parts[0];
String pass = parts.length > 1 ? parts[1] : null;
String credentials = Credentials.basic(user, pass);
client.authenticator((route, response) -> {
return response.request().header("Authorization") != null ? null : response.request().newBuilder().header("Authorization", credentials).build();
});
}
client.addNetworkInterceptor((chain) -> {
Request request = chain.request();
Response response = chain.proceed(request);
return response.code() == 408 ? response.newBuilder().code(500).message("Server-Side Timeout").build() : response;
});
return new OkHttpClient(client.build(), url);
}
};
}
@Override
public void cleanupIdleClients() {
this.pool.evictAll();
}
以下是如何把自定义的HttpCommandExecutor赋给Webdriver,到此即可
private static WebDriver startRemoteWebDriver(URL url, DesiredCapabilities capabilities) {
HttpCommandExecutor httpCommandExecutor = new HttpCommandExecutor(ImmutableMap.of(), url, new HttpClientFactory());
return new RemoteWebDriver(httpCommandExecutor, capabilities);
}
该方式也适用于基于appium技术栈的手机UI自动化~