[TOC]
自动化测试框架 Robotium的简单使用
- 待测试项目在 RobotiumDemo/ 中,AS向导中的LoginActivity,两个输入框加一个登录按钮而已;
- 修改
Build
--Select Build Variant
:- 我这里要在真机上运行,看到效果,选择
Test Artifact
为Android Instrumenttation Tests
; - 设定模块app的
Build Variant
为debug
;
- 我这里要在真机上运行,看到效果,选择
run
--Edit configurations
,在打开的Run/Debug Configuration
对话框中删除Junit
(若存在) 不然报错 [NoClassDefFoundError: junit/textui/ResultPrinter]( # http://stackoverflow.com/questions/19516289/exception-in-thread-main-java-lang-noclassdeffounderror-junit-textui-resultpr ):
java.lang.NoClassDefFoundError: junit/textui/ResultPrinter复制代码
导入
- jar包,放到AS工程模块
app/libs/
目录下,之后右键Add as Library...
即可,可以发现在app/build.gradle
中多了一项:
compile files('libs/robotium-solo-5.6.0.jar')复制代码
- 直接使用gradle导入线上版本:
# app/build.gradledependencies { compile fileTree(dir: 'libs', include: ['*.jar']) // testCompile 'junit:junit:4.12' ... testCompile 'com.jayway.android.robotium:robotium-solo:5.6.0'}复制代码
白盒测试
- 有源码时,可以直接创建测试用例,便可直接访问Activity,资源等;
- 在要测试的页面右键,
Go to
--Test(ctrl+shit+t)
--Create new test...
即可创建; - 手动继承
ActivityInstrumentationTestCase2
; - 创建无参构造函数,否则会报错:
junit.framework.AssertionFailedError: Class org.lynxz.robotiumdemoapp.LoginActivityTest has no public constructor TestCase(String name) or TestCase()复制代码
- 重写
setUp()
方法,初始化Solo对象,各种操作基本都是通过Solo来完成
Solo solo = new Solo(getInstrumentation(), getActivity());复制代码
- 创建测试方法:
// 名称任意 public void func_name() throws Exception { }复制代码
- 右键直接运行测试类即可;
- 常用方法:
// 查找指定id的控件 EditText edtMail = (EditText) mSolo.getView(R.id.email); // 输入文本 mSolo.enterText(edtMail, "lucid_lynxz@163.com"); // 清除文本 mSolo.clearEditText(edtMail); // 点击按钮(clickOnButton不太好用...) mSolo.clickOnView(btn); // 休眠一段时间,毫秒 mSolo.sleep(1000); // 截屏,储存到 /sdcard/Robotium-Screenshots/,可以带参数,指定名字 mSolo.takeScreenshot();复制代码
黑盒测试
这里有两篇资源:
步骤:
- 下载apk,重新签名为debug签名, 给出的小工具地址是: http://www.troido.de/re-sign.jar ,下不来就到 下; P.S.
- 使用方法也简单: 运行
re-sign.jar
后,将下载下来的apk拖动进去即可; 若是提示找不到找不到zipalign CreateProcess error=2
, 就从androidSDK/build-tools/
目录下复制一个zipalign
到提示目录里即可; - 有些apk会对签名文件进行检查或者使用其对资源文件进行加密,导致重新签名后无法运行,不过我是黑盒测试自己的应用(因为代码要使用公司的平台进行打包,会有写不同,所以黑盒测试),不考虑这个问题;
- 安装重新签名的apk到手机上(有源码可以直接运行源码,就会安装debug版本了);
- 新建一个工程,修改
app/build.gradle
,是工程包名与待测apk包名一致:
android { defaultConfig { applicationId "org.lynxz.robotiumdemoapp" # 改成待测apk包名 }}复制代码
- 按照白盒测试那样,新建测试类(在
androidTest/
目录下,Build Variants
配置同白盒一样),不过由于没有源码,因此对应的Activity需要通过反射来获取:
public class LoginActivityTest extends ActivityInstrumentationTestCase2 { public static final String PKG_NAME = "org.lynxz.robotiumdemoapp"; public static final String ACTIVITY_NAME = "org.lynxz.robotiumdemoapp.LoginActivity"; public LoginActivityTest() throws ClassNotFoundException { super(Class.forName(ACTIVITY_NAME)); } }复制代码
- 测试用例中,查找控件就不能使用常规的
R.id.***
了,可以通过text或者包名:id/***
的形式来指定:
// 实际的测试内容 public void testAutoLogin() throws Exception { // 查找控件,以下两种方式都可以,这里不能直接使用 'R.id.**' 来查找 final EditText edtMail = (EditText) mSolo.getView("email"); final EditText edtPwd = (EditText) mSolo.getView(PKG_NAME + ":id/password"); final Button btn = (Button) mSolo.getView(PKG_NAME + ":id/email_sign_in_button"); //对控件的操作(如输入文本,点击等)跟白盒测试一致 }复制代码
- 运行测试工程(重点) 这里就不能直接右键运行test项目了,因为会把
src/main
和src/androidTest
中的代码都打包安装,由于包名签名一致,main生成的apk会替换实际待测apk,而main中并无待测apk源码,导致test用例找不到所需页面/控件报错; 所以这里得单独运行androidTest.apk
,并手动启动它,方法就是在AS的Terminal
窗口(当然也可以使用系统的命令行窗口) 输入如下指令:
gradlew cleangradlew assembleDebugAndroidTest # 生成test的apk包adb install -r app/build/outputs/apk/app-debug-androidTest-unaligned.apk # 安装测试用例包adb shell am instrument -w your_pkg_name.test/android.test.InstrumentationTestRunner # 运行复制代码
最后一步是运行指定的testRunner,有可能会报错:
INSTRUMENTATION_STATUS: Error=Unable to find instrumentation info for: ComponentInfo
说明testRunner没有安装成功或者名字有写错,可以通过命令来查看已安装的runner:
adb shell pm list instrumentation复制代码
monkey
还没单独写monkey的,直接补充在这里了,记录下碰到的问题
E/hierarchyviewer: Unable to get view server version from device a5544648
在添加EasyMonkeyDevice
的时候,真机会报上述错误,代码如下:
from com.android.monkeyrunner import MonkeyRunner,MonkeyDevice from com.android.monkeyrunner import MonkeyImage from com.android.monkeyrunner.easy import EasyMonkeyDevice from com.android.monkeyrunner.easy import By device = MonkeyRunner.waitForConnection() easy_device = EasyMonkeyDevice(device) device.startActivity(component = 'com.nd.sdp.component.biz/com.nd.smartcan.appfactory.demo.SplashActivity') easy_device.touch(By.id('id/btn_login'),MonkeyDevice.DOWN_AND_UP)复制代码
找半天,在 找到了解释,原来需要真机中开启一个View Server的客户端与其进行socket通信,而在商业手机上,是无法开启view server的. 可以通过以下命令来查看/开关View server(我在自己未root的小米note上测试开启不成功):
adb shell service call window 1 i32 4939 # 开启命令,用此命令开启 adb shell service call window 2 i32 4939 # 关闭命令 adb shell service call window 3 # 查看状态 # 若返回值是:Result: Parcel(00000000 00000001 '........') 说明View Server处于开启状态 # 若返回值是:Result: Parcel(00000000 00000000 '........') 说明View Server处于关闭状态复制代码
又发现篇开启 View server
的 ,不过我没去折腾了; P.S. 使用 androidSDK/tools/monkeyrunner.bat your_python_script.py
来运行脚本;