悠悠小仙仙 发表于 2018-1-16 16:58:28

史上最简单Robotium跨进程操作实践——基于ADB框架

 Robotium是Android平台下一款非常优秀的自动化测试框架,它做android平台自动化的优势想必看到这篇文章的人应该都很清楚。但优点归优点,缺点也比较多,最明显的缺点有两个,一是必须
  Robotium是Android平台下一款非常优秀的自动化测试框架,它做android平台自动化的优势想必看到这篇文章的人应该都很清楚。但优点归优点,缺点也比较多,最明显的缺点有两个,一是必须要和被测系统签名保持一致,二是不能做任何跨进程的操作。
  很多小伙伴知道怎么用robotium,也知道它有这些限制,但不知道其中的原因。这里简单地说一下,robotium的这些先天不足的主要原因是由于它本身是基于instrumentation机制的,这既有好处也有坏处。好处是通过instrumentation注入到被测进程,从而与被测进程运行在同一进程空间,使得它能够非常方便地识别被测应用中的被测对象,并对这些对象进行操作。坏处是既然robotium已经跟被测应用"合体"了,那么根据android的进程隔离机制,它自然也被系统隔离在其他进程之外,无法跨进程操作任何对象,如图(1)所示。
  
       图(1)Android进程沙箱示意图
  其实使用相同的签名对于单应用的测试来说并不是难事,网上有很多应用重签名的方法和工具供大家使用。但不能跨进程操作确实成为了robotium最大的软肋,很多第三方的应用测试或多或少都有一些跨进程操作的测试场景,所以不少人因为这个问题而放弃了robotium,甚是可惜啊。
  作为robotium的铁杆粉之一(我最开始接触android自动化测试就是用的robotium),断断续续用了好几年,一直觉得是android平台最好用的自动化测试工具之一。对于robotium怎样突破进程的限制也做过一些研究,网上比较多的解决方案无外乎有以下几种:
  1.自己写服务做server,基于AIDL或编写socket与monkeyserver进行通讯,然后在robotium测试脚本里调用接口方法来间接地进行跨进程的操作,文章的作者只是给出了思路,我自己曾经按照这种方法去实现过,发现该方法的优势是比较稳定,缺点就是实现起来确实比较复杂,而且有些操作无法找到现成的系统aidl接口来进行操作,比如调用拍照操作,所以其实例子并不是很多,技术上的局限性较大。
  2.基于广播和service服务。这种方法我没有亲自去试过,但不难发现这种方法原理稍微简单些,但该方法貌似需要系统签名,而且自己写广播和服务来调用系统api,难度也不算小,所以一般不建议使用这种方式。
  综合上面的这些跨进程的解决方案,其实都感觉不太理想,对于很多初学者来说不是太难实现就是局限性比较大,一直没有找到一个比较理想的解决方案。直到后来在testerhome上看到一个技术很牛的兄弟把常用的adb命令做了一个很完备的封装(PS:真的是很全了,至少常用的adb命令都在,我甚至之前都没有想过adb命令还能做这么多事情),做成了一个独立的测试辅助工具。我就突然联想到为何不能使用adb命令来辅助robotium进行跨进程操作呢?既然该框架已经封装了基于adb的所有操作,而且adb又是不受系统限制的,那么基于这个框架理论上进行跨进程操作是没有任何问题的。后来经过试验,确实效果不错,轻量级、操作简单、使用方便,并且轻松跨进程,真是居家测试、屌丝逆袭的必备神器啊。好了,闲话不多说了,接下来我们就来详细地看看究竟是怎么用的。
  先来简单地介绍下这个adb命令框架的常用接口。这个框架里有三个主要的包,分别介绍如下:
  xuxu.autotest这个包里面主要有两个类,一个是AdbDevice,封装了功能测试中常用的一些操作,如获取当前activity的名称和包名、获取设备分辨率、关闭应用、点击对象等常用功能,非常实用。另一个类是XuImage,顾名思义,即封装了一些常见的图片操作,如得到指定边界的图片截图、对比图片是否一致、截取图片等。
  xuxu.autotest.element这个包主要用于获取被测对象,主要是封装了一个Position对象,用于通过ClassName、Id、Contentdesc等属性来获取一个被测对象,其底层是通过uiautomator来dump当前ui的xml文件的,可以获取到所有对象节点的xml文件。
  xuxu.autotest.utils这个包里面主要提供了对Date日期时间的操作,ImageUtil对图片的操作及正则表达式和shell语句的操作的封装,方便大家的使用。总的来说整个框架的接口设计还是非常齐全的,可以实现很多常用功能,具体功能大家可以自己看源码和帮助文档来摸索。
  接下来就进入我们本次的主题--跨进程。为了方便大家理解,我会在本文中选择两个非常常见的跨进程操作场景来说明如何跨进程,也就是大家喜闻乐见的相机拍照和打电话来进行演示。
  例子1、跨进程操作之相机拍照。
  被测程序非常简单,程序界面截图如下图(2)所示:
  
  点击第一个界面的"拍照片"按钮后,进入界面2,点击其中的"拍张照片"后启动系统的相机,当用户按下拍照功能键后,系统可以将用户拍下的照片显示在应用中,以备后续浏览或者是上传。由于相机应用和我们所写的被测应用是两个不同的应用,所以这就属于典型的跨进程操作,Robotium框架本身是无法对这个相机界面进行操作的。
  接下来看看关键的测试工程怎么创建的。测试工程的创建方法非常简单,操作步骤如下:
  1.首先按照常规的android测试工程的建法,创建一个常规测试工程。
  2.在工程中引入robotium和adbForAndroid的jar包。
  这样,我们的测试工程就准备好了。
  接下来的步骤就是写测试脚本。由于AdbForAndroid框架是按照元素的相关属性来查找并定位被测对象的,所以首先要弄清楚我们要操作的跨进程的界面上的对象的信息,这些信息我们可以通过很多现成工具来看,我这里选择android自带的uiautomatorviewer。先在模拟器上手工打开被测程序,进入拍照的相机界面,并用uiautomatorviewer查看界面元素信息如下图(3)所示:
  
  我们可以看到,我们要点击的拍照按钮的属性在图(3)的右下角的列表中已全部展现出来。那么我们该选哪个呢?这个问题取决于adbForAndroid框架支持由哪些属性来获取元素。通过查询其帮助文档,我们知道目前这个框架中常用的查找元素的方法如下:
  findElementByContentdesc
  findElementByClass
  findElementByText
  findElementById
  由于我现在使用的模拟器版本为4.2.2,所以uiautomatorviewer无法显示id属性,如果大家用的是4.3版本以上,就可以看到元素的id属性了。
  那么现在这种情况来看,我们最好就选择Contentdesc属性来定位对象了。代码很简单,如下:
  viewplaincopyElementelement=position.findElementByContentdesc("Shutterbutton");
  adbDevice.tap(element);//点击拍照按钮
  好,接下来点击了拍照按钮之后,拍照功能还会让你选择是确定还是取消操作,如下图(4)所示:


悠悠小仙仙 发表于 2018-1-16 17:01:20

  那么如法炮制,我们通过查询获知,这个用于确定的"√"按钮,它的属性能用于定位的其实只有class属性。不过这里要注意的是,此时由于我们用的是class属性,大家可以看到界面上class属性跟我们要点击的"√"这个按钮相同的对象很多,所以我们必须用findElementsByClass方法了。这个方法返回的是一个ArrayList,所以我们可以写如下的代码来获取所有class属性为"android.widget.ImageView"的元素列表。
  viewplaincopyArrayListimageViews=position.findElementsByClass("android.widget.ImageView");
  好,写完之后,现在问题来了,究竟这个数组里面哪个index才是对应的我们要点击的"√"按钮呢?经过试验,我发现界面上多个相同元素返回到数组中时,对应元素位置是按照界面上的位置从上到下、从左到右来的,所以我们要点击的这个按钮的index应该是4。所以点击它的代码如下:
  viewplaincopyadbDevice.tap(imageViews.get(4));//2是x,3是重拍,4是√
  上面的代码运行完后,界面就会回到我们的被测程序,后面的操作就不用我再多说了,大家看看是不是非常简单?
  例子2、跨进程操作之打电话
  有了第一个例子的基础,其实第二个例子就很好实现了。被测程序非常简单,如下图(5)所示:
 
  点击"拨打该号码"后,系统自动进入拨号界面,所以也是典型的跨进程测试场景。
  实现方法还是跟例1中一样,先还是用uiautomatorviewer查看界面中的对象信息,再使用对应的方法来操作对象即可。而且在这个例子中,我还给大家演示了另外一种情况,即有些跨进程操作不但要进行操作,还要取得一些对象属性来进行验证,这也是基本可以的。这里我直接给出我的测试代码(稍微封装了一下):
  viewplaincopypublicbooleanCallUtil(StringcallNumber){
  Elementelement;
  booleanresult;
  //验证是否拨打了正确的号码
  if(callNumber.length()==11){//正常号码需要转变为xxxx-xxx-xxxx的格式
  StringformatCallNumber=callNumber.substring(0,1)+""+callNumber.substring(1,4)+"-"+callNumber.substring(4,7)+"-"+callNumber.substring(7,callNumber.length());
  element=position.findElementByText(formatCallNumber);
  try{
  Thread.sleep(2000);//线程休眠2秒
  }catch(InterruptedExceptione){
  e.printStackTrace();
  }
  }else{
  element=position.findElementByText(callNumber);//除正常号码外,其他格式的号码不进行格式转换
  }
  if(element!=null){
  result=true;
  }else{
  result=false;
  }
  element=position.findElementByContentdesc("End");//挂断电话
  if(element!=null)adbDevice.tap(element);
  returnresult;
  }
  好了,整个过程非常简单。我相信能用robotium的童鞋用起来应该都没有任何问题,所有代码都是非常易用易懂的。
  相信通过前面的实例,大家可以发现说这是"史上最简单"的Robotium跨进程操作解决方案我估计没人会反对,这并不是夸大事实、博人眼球,确实很简单,功能也很强大。其他也不用我多说了,最后我再来对该框架总结一下下吧:
  优点:
  1.确实非常全,基本封装了adb的所有常用命令,它本身就是一个adb命令使用大全了。
  2.框架接口设计清晰易懂,简单明了,封装成jar包,用起来也很方便。另外希望大家都能看看它的源码并了解它的实现细节,作者封装得还是很不错的,很值得我们学习,而不仅仅是简单地使用它。
  3.不需要root权限。其实这点还是很重要的,有些公司的测试机是没有root权限,用adb就很方便。对于不想放弃robotium,又确实有跨进程操作需求的童鞋来说,真的是非常实用的一个框架。
  目前已知的不足:
  1.手机的版本必须是4.1以上,即至少必须支持uiautomator的手机才行,因为该框架本身底层是依赖于uiautomator来dump出对象布局xml文件,最终获取对象的坐标进行操作的,所以你的手机版本本身并不支持uiautomator的话,就没法实现dump操作。
  2.有些对象是uiautomator也无法识别和操作的,那当然这个框架也无能为力了。比如屏幕最上方的通知消息栏的对象,所有工具都没法显示和识别,这个肯定没办法了,另外还比如输入文字时的弹出键盘上的按钮对象等,也无法识别,大家可以试试。
  以后想要尝试的改进:
  1.对于手机版本的支持问题,个人觉得有个曲线救国的方法可以解决。即我们可以稍微修改下源码,加入判断手机版本的代码,判断当前手机版本如果高于4.1,就直接通过uiautomator来dump,如果低于4.1,则读取PC上事先导出到指定位置的xml文件。这样的话,如果你使用的手机不是4.1以上的版本,只需要事先把被测应用用4.1以上版本的手机通过uiautomator先dump到PC的一个指定路径上就行了,这样效果应该是一样的,只要最终能得到对象坐标就OK。
  2.再加一些比较实用的功能。比如现在可以根据text来查找对象,但实际上很多时候我们可能是得到对象了,但想通过这个对象去获取对象的其他属性,所以建议增加类似getXXXXByElement(Elemente)这样的方法,有空试一下,呵呵。
  以上只是我的一些个人想法,如果大家还有一些什么别的建议,也欢迎大家都提出来,一起来完善这个很实用的框架。
页: [1]
查看完整版本: 史上最简单Robotium跨进程操作实践——基于ADB框架