文章出处http://blog.csdn.net/jiuzuidongpo/article/details/51790455
Appium在接收到客户端脚本的连接之后的初始化准备工作列表(细节部分详细叙述,只说重点):
下面提到的有关目录是自己的本地目录,需要注意。
1,检查app包是否存在,检查adb工具是否可用,检查java工具包版本。
2,确定设备,用命令
adb.exe devices
选择指定udid的设备
3,确定设备是否准备好,命令
adb.exe -s 16144573 wait-for-device
adb.exe -s 16144573 shell "echo 'ready'"
如果命令行输出ready,则表示准备好
4,检查设备的API版本号,命令
adb.exe -s 16144573 shell "getprop ro.build.version.sdk"
如果>=17,则可用
5,语言设置
获取设备当前使用语言(一般返回zh)
adb.exe -s 16144573 shell "getprop persist.sys.language"
生成string.json
java -jar "C:Program Files (x86)Appium ode_modulesappium ode_modulesappium-adbjarsappium_apk_tools.jar" "stringsFromApk" "E:workspaceappium-testsampleappsContactManagerContactManager.apk" "C:UsersADMINI~1AppDataLocalTempcom.example.android.contactmanager" zh
如果报错,则执行下面,生成string.json
java -jar "C:Program Files (x86)Appium ode_modulesappium ode_modulesappium-adbjarsappium_apk_tools.jar" "stringsFromApk" "E:workspaceappium-testsampleappsContactManagerContactManager.apk" "C:UsersADMINI~1AppDataLocalTempcom.example.android.contactmanager"
然后从string.json设置成默认语言
D:softandroid-sdk-windowsplatform-toolsadb.exe -s 16144573 push "C:\Users\ADMINI~1\AppData\Local\Temp\com.example.android.contactmanager\strings.json" /data/local/tmp
6,检查aapt工具,并且用aapt工具分析apk包。
D:softandroid-sdk-windowsuild-tools23.0.1aapt.exe dump xmltree E:workspaceappium-testsampleappsContactManagerContactManager.apk AndroidManifest.xml
获取到apk的进程名称:com.example.android.contactmanager,(如果指定的话,则没必要获取)。
和开始的activity:.ContactManager,(如果指定的话,没必要去获取)。
7,注册认证apk包
java -jar "C:Program Files (x86)Appium ode_modulesappium ode_modulesappium-adbjarsverify.jar" E:workspaceappium-testsampleappsContactManagerContactManager.apk
8,压缩apk包(Zip-aligning),并且放入手机中,这个过程有点儿复杂。
D:softandroid-sdk-windowsuild-tools23.0.1zipalign.exe -f 4 E:workspaceappium-testsampleappsContactManagerContactManager.apk C:UsersADMINI~1AppDataLocalTemp116530-6572-vjd4buappium.tmp
获取MD5
MD5 for app is b2d2916bb5388e1dc281ec3e71ef1234
查看apk是否存在
D:softandroid-sdk-windowsplatform-toolsadb.exe -s 16144573 shell "ls /data/local/tmp/b2d2916bb5388e1dc281ec3e71ef1234.apk"
查看apk是否安装
D:softandroid-sdk-windowsplatform-toolsadb.exe -s 16144573 shell "pm list packages -3 com.example.android.contactmanager"
创建文件夹
D:softandroid-sdk-windowsplatform-toolsadb.exe -s 16144573 shell "mkdir -p /data/local/tmp/"
查看存在的apk
D:softandroid-sdk-windowsplatform-toolsadb.exe -s 16144573 shell "ls /data/local/tmp/*.apk"
删除存在的apk
D:softandroid-sdk-windowsplatform-toolsadb.exe -s 16144573 shell rm "/data/local/tmp/29649242b53e9a67ba855b067422713c.apk"
放入手机中
D:softandroid-sdk-windowsplatform-toolsadb.exe -s 16144573 push "E:\workspace\appium-test\sample\apps\ContactManager\ContactManager.apk" /data/local/tmp/b2d2916bb5388e1dc281ec3e71ef1234.apk
停止之前运行的
D:softandroid-sdk-windowsplatform-toolsadb.exe -s 16144573 shell "am force-stop com.example.android.contactmanager"
D:softandroid-sdk-windowsplatform-toolsadb.exe -s 16144573 shell "pm clear com.example.android.contactmanager"
卸载之前运行的
D:softandroid-sdk-windowsplatform-toolsadb.exe -s 16144573 uninstall com.example.android.contactmanager
安装新apk包
D:softandroid-sdk-windowsplatform-toolsadb.exe -s 16144573 shell "pm install -r /data/local/tmp/b2d2916bb5388e1dc281ec3e71ef1234.apk"
9,绑定本机和boostrap通信的端口号
D:softandroid-sdk-windowsplatform-toolsadb.exe -s 16144573 forward tcp:4724 tcp:4724
10,将bootstrap.jar放入手机中
D:softandroid-sdk-windowsplatform-toolsadb.exe -s 16144573 push "C:\Program Files (x86)\Appium\node_modules\appium\build\android_bootstrap\AppiumBootstrap.jar" /data/local/tmp/
11,让Unicode键盘可用
将Unicode输入法push到手机中
D:softandroid-sdk-windowsplatform-toolsadb.exe -s 16144573 install "C:Program Files (x86)Appium ode_modulesappiumuildunicode_ime_apkUnicodeIME-debug.apk"
获取手机当前的输入法,测试之后要恢复这个输入法
D:softandroid-sdk-windowsplatform-toolsadb.exe -s 16144573 shell "settings get secure default_input_method"
设置unicode键盘可用,并且为当前系统的默认输入法
D:softandroid-sdk-windowsplatform-toolsadb.exe -s 16144573 shell "ime enable io.appium.android.ime/.UnicodeIME"
D:softandroid-sdk-windowsplatform-toolsadb.exe -s 16144573 shell "ime set io.appium.android.ime/.UnicodeIME"
12,安装appiium的setting和unlock测试包
unlock
D:softandroid-sdk-windowsplatform-toolsadb.exe -s 16144573 install "C:Program Files (x86)Appium ode_modulesappiumuildunlock_apkunlock_apk-debug.apk"
Setting
D:softandroid-sdk-windowsplatform-toolsadb.exe -s 16144573 install "C:Program Files (x86)Appium ode_modulesappiumuildsettings_apksettings_apk-debug.apk"
13,启动手机上的bootstrap。
首先停止手机上之前的boostrap:
获取当前运行的boostrap
D:softandroid-sdk-windowsplatform-toolsadb.exe -s 16144573 shell "ps 'uiautomator'"
获取到进程号16324,然后杀死
D:softandroid-sdk-windowsplatform-toolsadb.exe -s 16144573 shell "kill 16324"
然后开始当前的boostrap
D:softandroid-sdk-windowsplatform-toolsadb.exe -s 16144573 shell uiautomator runtest AppiumBootstrap.jar -c io.appium.android.bootstrap.Bootstrap -e pkg com.example.android.contactmanager -e disableAndroidWatchers false
14,然后就可以用socket和手机通信了,通信的前三步
第一步:
向手机端发送:{"cmd":"action","action":"wake","params":{}}
使手机觉醒,并且判断锁屏的情况(这个有待研究),如果锁屏,运行unlock可以解锁
D:softandroid-sdk-windowsplatform-toolsadb.exe -s NVY9AUSO5SAELRSC shell "am start -S -a android.intent.action.MAIN -c android.intent.category.LAUNCHER -f 0x10200000 -n io.appium.unlock/.Unlock"
第二步:
向手机发送:{"cmd":"action","action":"getDataDir","params":{}}
获取数据存放路径,正常返回{"value":"/data/local/tmp","status":0},好像没啥用
再发送
{"cmd":"action","action":"compressedLayoutHierarchy","params":{"compressLayout":false}}
返回{"value":false,"status":0}
是某个支持项。
第三步:
启动要测试的apk
D:softandroid-sdk-windowsplatform-toolsadb.exe -s NVY9AUSO5SAELRSC shell "am start -S -a android.intent.action.MAIN -c android.intent.category.LAUNCHER -f 0x10200000 -n com.example.android.contactmanager/.ContactManager"
让app获得当前的焦点
D:softandroid-sdk-windowsplatform-toolsadb.exe -s NVY9AUSO5SAELRSC shell "dumpsys window windows"
- 1. 建立session时常用命令:
DesiredCapabilities cap = new DesiredCapabilities();
cap.SetCapability("browserName", ""); // web 浏览器名称('Safari' ,'Chrome'等)。如果对应用进行自动化测试,这个关键字的值应为空。
cap.SetCapability("platformName", "Android");//你要测试的手机操作系统
cap.SetCapability("platformVersion", "4.4");//手机操作系统版本
cap.SetCapability("automationName", "selendroid"); //你想使用的自动化测试引擎:Appium (默认) 或 Selendroid
cap.SetCapability("deviceName", " Android Emulator"); //使用的手机类型或模拟器类型,真机时输入Android Emulator或者手机型号
cap.SetCapability("udid", udid); //连接的物理设备的唯一设备标识,Android可以不设置
cap.SetCapability("newCommandTimeout", "300"); //设置收到下一条命令的超时时间,超时appium会自动关闭session ,默认60秒
cap.SetCapability("unicodeKeyboard", "True");//支持中文输入,会自动安装Unicode 输入法。默认值为 false
cap.SetCapability("resetKeyboard", "True"); //在设定了 unicodeKeyboard 关键字的 Unicode 测试结束后,重置输入法到原有状态
cap.SetCapability("'app'", "D:\AndroidAutomation\AndroidAutoTest\app\zhongchou.apk"); //未安装应用时,设置app的路径
//手机已安装app,直接从手机启动app,上面路径不设置
cap.SetCapability("appPackage", "com.nbbank"); //你要启动的Android 应用对应的Activity名称|比如`MainActivity`, `.Settings`|
cap.SetCapability("appActivity", "com.nbbank.ui.ActivityShow"); //你想运行的Android应用的包名
cap.SetCapability("appWaitActivity", "com.nbbank.ui.ActivityLogo"); //你想要等待启动的Android Activity名称|比如`SplashActivity`|
Uri serverUri = new Uri("http://127.0.0.1:4723/wd/hub");
driver = new AndroidDriver<IWebElement>(serverUri, cap, TimeSpan.FromSeconds(180));
更多详细查看官网:https://github.com/appium/appium/blob/master/docs/cn/writing-running-appium/caps.cn.md
- 2. driver常用方法及注意事项
1) 常用方法:
driver.HideKeyboard();//隐藏键盘
driver.BackgroundApp(60);//60秒后把当前应用放到后台去
driver.LockDevice(3); //锁定屏幕
//在当前应用中打开一个 activity 或者启动一个新应用并打开一个 activity
driver.StartActivity("com.iwobanas.screenrecorder.pro", "com.iwobanas.screenrecorder.RecorderActivity");
driver.OpenNotifications();//打开下拉通知栏 只能在 Android 上使用
driver.IsAppInstalled("com.example.android.apis-");//检查应用是否已经安装
driver.InstallApp("path/to/my.apk");//安装应用到设备中去
driver.RemoveApp("com.example.android.apis");//从设备中删除一个应用
driver.ShakeDevice();//模拟设备摇晃
driver.CloseApp();//关闭应用
driver.LaunchApp();//根据服务关键字 (desired capabilities) 启动会话 (session) 。请注意这必须在设定 autoLaunch=false 关键字时才能生效。这不是用于启动指定的 app/activities
driver.ResetApp();//应用重置
driver.GetContexts();//列出所有的可用上下文
driver.GetContext();//列出当前上下文
driver.SetContext("name");//将上下文切换到默认上下文
driver.GetAppStrings();//获取应用的字符串
driver.KeyEvent(176);//给设备发送一个按键事件:keycode
driver.GetCurrentActivity();//获取当前 activity。只能在 Android 上使用
//driver.Pinch(25, 25);//捏屏幕 (双指往内移动来缩小屏幕)
//driver.Zoom(100, 200);//放大屏幕 (双指往外移动来放大屏幕)
driver.PullFile("Library/AddressBook/AddressBook.sqlitedb");//从设备中拉出文件
driver.PushFile("/data/local/tmp/file.txt", "some data for the file");//推送文件到设备中去
driver.FindElement(By.Name(""));
driver.FindElementById("id");
driver.FindElementByName("text");
driver.FindElementByXPath("//*[@name='62']");
2) 注意事项:
使用driver.Sendkeys(string str)向文本框输入内容前,最好先element.Click( )一下,否则某些情况下,输入的内容会请不掉,文本框提示的内容也会在 输入的文本前显示出来。sendkey方法在发送数据之前会清空一下文本框,一般不需要Clear,如前面的情况Clear后仍是存在的,click后正常
- 3. 等待页面加载策略:
1) 显性等待:调用selenium的方法, 需要添加WebDriver.Support引用
显性等待是指在代码进行下一步操作之前等待某一个条件的发生。最不好的情况是使用Thread.sleep()去设置一段确认的时间去等待。但为什么说最不好呢?因为一个元素的加载时间有长有短,你在设置sleep的时间之前要自己把握长短,太短容易超时,太长浪费时间。selenium webdriver提供了一些方法帮助我们等待正好需要等待的时间
WebDriverWait wait = new WebDriverWait(driver, TimeSpan.FromSeconds(10));
element = wait.Until<IWebElement>((d) =>
{
return driver.FindElement(By.Id("userName"));
});
2) 隐性等待:设置时间不易过长,设置为500或1000即可
隐性等待是指当要查找元素,而这个元素没有马上出现时,告诉WebDriver查询Dom一定时间。默认值是0,但是设置之后,这个时间将在WebDriver对象实例整个生命周期都起作用。
driver.Manage().Timeouts().ImplicitlyWait(TimeSpan.FromSeconds(1));
- 4. drive.KeyEvent(int )的使用: 可使用KeyEvent发送键盘数据,比如退格,Enter键等
driver.KeyEvent(3); //KEYCODE_HOME 按键Home 3
driver.KeyEvent(26); //KEYCODE_POWER 电源键 26
driver.KeyEvent(67); //KEYCODE_DEL 退格键 67
driver.KeyEvent(66); //KEYCODE_ENTER 回车键
driver.KeyEvent(122); //KEYCODE_MOVE_HOME 光标移动到开始
driver.KeyEvent(123); //KEYCODE_MOVE_END 光标移动到末尾
-
5. 坐标操作
为防止不同手机分辨率不同带来的影响,要避免使用固定的坐标,可以用以下方式获取元素的坐标
double Screen_X = driver.Manage().Window.Size.Width;//获取手机屏幕宽度
double Screen_Y = driver.Manage().Window.Size.Height;//获取手机屏幕高度
double startX = element.Location.X; //获取元素的起点坐标,即元素最左上角点的横坐标
double startY = element.Location.Y; //获取元素的起点坐标,即元素最左上角点的纵坐标
double elementWidth = element.Size.Width; //获取元素的宽度
double elementHight = element.Size.Height; //获取元素的宽度
在封装“滑动”、“ TouchAction”等操作时可以用以上方法来获取坐标进行操作。
示例:分装两个元素之间的滑动
IWebElement elmentA = null;
IWebElement elmentB = null;
int startX = 0, startY = 0, endX = 0, endY = 0;
int duration=0,time=0;
/// <summary>
/// 从元素A的位置滑动到元素B的位置
/// </summary>
/// <param name="A">元素A的名称</param>
/// <param name="B">元素B的名称</param>
/// <param name="sDuration">滑动持续时间</param>
/// <param name="sTime">滑动次数</param>
public void SwipeAToB(string A, string B,string sDuration,string sTime)
{
startX = elmentA.Location.X + elmentA.Size.Width / 2; //元素A的中心横坐标
startY = elmentA.Location.Y + elmentA.Size.Height / 2; //元素A的中心纵坐标
endX = elmentB.Location.X + elmentB.Size.Width / 2; //元素B的中心横坐标
endY = elmentB.Location.Y + elmentB.Size.Height / 2; //元素B的中心纵坐标
duration = string.IsNullOrEmpty(sDuration) ? 1500 : int.Parse(sDuration); //持续时间为空时,默认设置为1500毫秒
time = string.IsNullOrEmpty(sTime) ? 1500 : int.Parse(sTime); //滑动次数为空时,默认设置为滑动1次
for (int i = 0; i < time; i++)
{
driver.Swipe(startX, startY, endX, endY, duration);
}
}
注意:element.Loaction和element.Size,每次获取时都会重新去手机里获取,为节省时间如果有获取相同值的,建议储存成变量。
- 6. 取消重新安装unlock和setting
注销如下代码:
Appium ode_modulesappiumlibdevicesandroidandroid.js
async.series([
this.initJavaVersion.bind(this),
this.initAdb.bind(this),
this.packageAndLaunchActivityFromManifest.bind(this),
this.initUiautomator.bind(this),
this.prepareDevice.bind(this),
this.checkApiLevel.bind(this),
this.pushStrings.bind(this),
this.processFromManifest.bind(this),
this.uninstallApp.bind(this),
this.installAppForTest.bind(this),
this.forwardPort.bind(this),
this.pushAppium.bind(this),
this.initUnicode.bind(this),
// DO NOT push settings app and unlock app
//this.pushSettingsApp.bind(this),
//this.pushUnlock.bind(this),
function (cb) {this.uiautomator.start(cb);}.bind(this),
this.wakeUp.bind(this),
this.unlock.bind(this),
this.getDataDir.bind(this),
this.setupCompressedLayoutHierarchy.bind(this),
this.startAppUnderTest.bind(this),
this.initAutoWebview.bind(this),
this.setActualCapabilities.bind(this)
], function (err) {