针对智能手机端的应用的自动化测试,目前主要分两类:一是基于APP的apk自动化测试,二是浏览器的wap页测试。目前做的较多的是第一种情况,应用的自动化测试框架也较多,如NativeDriver、Robotium、calabash等;而第二种情况的自动化测试框架就较少了,目前较常用的为Selenium。
而利用Selenium的测试框架有Selenium+Junit、Selenium+testng为最常用。这里我就介绍下利用Selenium+testng测试手机端的wap页的页面展示。
该框架测试有个明显的缺点是:它利用AndroidDriver驱动,提供webview来显示http请求返回的内容,不能在其他浏览器上运行,就是说不能测试浏览器的兼容性。但这个可以测试不同机型的手机,做适配测试。因为webview使用android内置webkitt内核,不同机型的内核是有异的。
第一步,利用该框架测试手机端wap页时,必须在测试工程中(java project)导入selenium-server-standalone-2.25.0.jar和testng.jar。同时在手机端需要安装(webdriver)android-server-xxx.apk作为服务端分发http请求命令和显示返回内容。
第二步,在工程中,在准备好了上面的条件,可以通过命令来打开手机端的webdriver和连接服务端(启动手机端的webdriver)——放在测试用例执行前;当然也可以手动启动webdriver和在DOS下运行命令:
如
@BeforeTest public void setUp() { // 启动手机端的webdriver——作为服务端,没有其他前台应用 try { //打开webdriver Runtime.getRuntime().exec( "adb shell am start -a android.intent.action.MAIN " + "-n org.openqa.selenium.android.app/.MainActivity"); /*连接服务端webdriver,连接用的端口为8080,如果本地有开启了或使用了该端口,需要关闭(http://localhost:8080/exit),不然连接不上服务端webdriver */ Runtime.getRuntime().exec("adb forward tcp:8080 tcp:8080"); } catch (IOException e) { e.printStackTrace(); } driver = new AndroidDriver(); }
第三步,写测试用例,如
//用例中要用到外来数据时,作为参数queryString1传递,来源为dataProvider = “testdata”中的数据
@Test(dataProvider = "testdata") public void testQuery1(String queryString1) { //具体的用例执行(包括调用滑动和截屏,来达到获取想要的页面) }
第四步,用例执行完之后,需要关闭终止驱动
@AfterTest public void tearDown() { driver.quit(); }
二、关于testng用例中带参数的数据源获取,在testng中大概有三种方法:一种是在testng.xml里配置,用@Parameters引用;第二种是使用DataProvider;第三种是继承FeedTest类获取数据。
这里主要讲下,第二种和第三种的使用:
1、使用DataProvider时,会在用例方法前有@Test(dataProvider = “testdata”),如
@Test(dataProvider = "testdata") public void testQuery1(String queryString1) { try { // 统一编码格式 QUERY = URLEncoder.encode(queryString1.trim(), "utf-8"); } catch (UnsupportedEncodingException e1) { // TODO Auto-generated catch block e1.printStackTrace(); } //具体的用例执行 }
同时有个方法是获取数据源的,如以下例子,从txt文件中读取数据,并放在filelist中,为了DataProvider方便执行,返回列表的迭代器。读取数据时,有个编码encoding处理,这个是为了保证txt文件存储格式(UTF-8)一致,一遍用例执行使用的数据不会出现数据表示不一致的情况。当然为了任何地方一致,用例中对数据也最好处理,如上。
@DataProvider(name = "testdata") public Iterator<String[]> getData() { ArrayList<String[]> filelist = new ArrayList<String[]>(); String filepath = Data.getFileString(); try { String encoding = "UTF-8"; File file = new File(filepath); if (file.isFile() && file.exists()) { // 判断文件是否存在 InputStreamReader read = new InputStreamReader( new FileInputStream(file), encoding); // 考虑到编码格式 BufferedReader bufferedReader = new BufferedReader(read); String lineTxt = ""; while ((lineTxt = bufferedReader.readLine()) != null) { String[] str = new String[1]; str[0] = lineTxt.trim(); filelist.add(str); } read.close(); } else { System.out.println("找不到指定的文件"); } } catch (Exception e) { System.out.println("读取文件内容出错"); e.printStackTrace(); } return filelist.iterator(); }
2、使用FeedTest获取数据时,必须导入feed4testng-1.0-dist里的jar包才能使用,然后在测试类中继承FeedTest类就行,这个能自动读取数据,现在支持自动读取的有.xls和.cvs文件
如
public class OneTest extends FeedTest { @Test(dataProvider = "feeder") // 数据来源excel文件xls(97-2003版的) @Source("D:/case/testcase2.xls") public void testQuery1(String queryString1) { /* 参数分别是读取xls文件的第一列、、、,第一行数据读不出,那是因为默认数据字段名称 */ } }
使用该方式读取数据,我们可以不用再自己写方法读取数据了。但缺点就是数据来源的路径是固定的,如果在运行时要改变数据来源,是不行的。
除了支持两种文件可以@Source自动读取,可以读取其他类型的文件和数据库,具体可参见:
http://databene.org/feed4testng.html
三、针对自动化测试wap页面展示问题,当要测试的展示页面是通过在页面输入一个搜索词query的返回结果时,且query较多。在这种情况下我们手工输入再看页面展示,工作量就较大了。所以需要自动化测试(利用selenium),这样我们就不需要手动输入query再滑动页面看页面展示了;只需要看截下来的页面展示图就可以了 。那我们要实现的自动化就有搜索词展示、滑动页面到能看到的视区内、截屏。
- 针对搜索词query,有个注意的地方就是:搜索词不能有中文,如 element.sendKeys(“比亚迪”);就是不行,webdriver调起输入法时,只有英文输入那个界面。解决这个问题的方法就是把搜索词query放在URL里,如driver.get(“http://m.so.com/s?q=”+”比亚迪”);
- 针对搜索词的结果页,当前屏幕显示的不一定是需要的展示,而我们要截屏,又只能截取当前显示的,那么就只能滑动页面到当前显示的界面。
滑动有两种方法:两种方法都依赖于定位的元素element
如:下面的代码表示我想找测试类别为“汽车”的搜索词的某种展示,结果显示条数>=1。
// 该方法可以根据测试要求改变,(定位到什么元素) public WebElement scrollCondition(WebDriver driver, String queryclass) { // 寻找当前显示是否到达要求的显示,不是的话滑动 if (!queryclass.isEmpty()) { if (queryclass.contains("汽车")) { // 汽车类 elements = driver.findElements(By .xpath("//h1 [@class='car-title icon-mobile']")); } } if ((elements != null) && (elements.size() > 0)) { element = elements.get(0); } }
方法一:
//该方法滑动后不停留,不利于截屏,不建议使用 Actions action=new Actions(driver); action.moveToElement(element).perform();
方法二:
((JavascriptExecutor) driver).executeScript("arguments[0].scrollIntoView();", element);
关于截屏可以利用ddmlib来截取(需要ddmlib.jar)
http://blogs.360.cn/360qtest/2014/02/28/通过ddmlib实现截图以及获取手机相关信息/