appium的初始化准备工作

文章出处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) {
原文地址:https://www.cnblogs.com/111testing/p/7824065.html