群控编程-Appium 学习 总结 (C#)

admin   ·   发表于 2018-7-2   ·   文章分享


Guide

From  https://testerhome.com/topics/3306



最近看到测试群里小白们经常问的问题,在此写一下自己学习的一点小总结

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 (默认) 或 Selendroidcap.SetCapability("deviceName", " Android Emulator"); //使用的手机类型或模拟器类型,真机时输入Android Emulator或者手机型号cap.SetCapability("udid", udid); //连接的物理设备的唯一设备标识,Android可以不设置cap.SetCapability("newCommandTimeout", "300");  //设置收到下一条命令的超时时间,超时appium会自动关闭session ,默认60秒cap.SetCapability("unicodeKeyboard", "True");//支持中文输入,会自动安装Unicode 输入法。默认值为 falsecap.SetCapability("resetKeyboard", "True"); //在设定了 unicodeKeyboard 关键字的 Unicode 测试结束后,重置输入法到原有状态cap.SetCapability("'app'", "D:\\AndroidAutomation\\AndroidAutoTest\\app\\zhongchou.apk");  //未安装应用时,设置app的路径//手机已安装app,直接从手机启动app,上面路径不设置cap.SetCapability("appPackage", "com.xxx");  //你要启动的Android 应用对应的Activity名称|比如`MainActivity`, `.Settings`|cap.SetCapability("appActivity", "com.xxx.ui.ActivityShow");  //你想运行的Android应用的包名cap.SetCapability("appWaitActivity", "com.xxx.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 或者启动一个新应用并打开一个 activitydriver.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/activitiesdriver.ResetApp();//应用重置driver.GetContexts();//列出所有的可用上下文driver.GetContext();//列出当前上下文driver.SetContext("name");//将上下文切换到默认上下文driver.GetAppStrings();//获取应用的字符串driver.KeyEvent(176);//给设备发送一个按键事件:keycodedriver.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键等,注意:Androiddriver 使用 driver.PressKeyCode( int keycode)

driver.KeyEvent(3); //KEYCODE_HOME 按键Home 3driver.KeyEvent(4); //KEYCODE_BACK  返回键   4driver.KeyEvent(82); //KEYCODE_MENU 菜单键   82driver.KeyEvent(26);  //KEYCODE_POWER 电源键 26driver.KeyEvent(24);  //KEYCODE_VOLUME_UP   音量增加键 24driver.KeyEvent(25);  //KEYCODE_VOLUME_DOWN 音量减小键 25driver.KeyEvent(67);  //KEYCODE_DEL 退格键 67driver.KeyEvent(66);  //KEYCODE_ENTER 回车键driver.KeyEvent(122); //KEYCODE_MOVE_HOME 光标移动到开始driver.KeyEvent(123); //KEYCODE_MOVE_END 光标移动到末尾

keycode 参考:Android KeyCode列表

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. 关闭自动推送appium bootstrap和安装unlock和setting

一般测试过一次,不需要再重新安装,重新安装比较浪费时间,且有些安装时会有弹窗
注销如下代码:

Appium\node_modules\appium\lib\devices\android\android.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) {

7. 取消重新安装UnicodeIME输入法

修改如下代码:

Appium\node_modules\appium\lib\devices\android\android-common.js

androidCommon.pushUnicodeIME = function (cb) {
  cb()
  /*
  logger.debug("Pushing unicode ime to device...");
  var imePath = path.resolve(__dirname, "..", "..", "..", "build",
      "unicode_ime_apk", "UnicodeIME-debug.apk");
  fs.stat(imePath, function (err) {
    if (err) {
      cb(new Error("Could not find Unicode IME apk; please run " +
                   "'reset.sh --android' to build it."));
    } else {
      this.adb.install(imePath, false, cb);
    }
  }.bind(this));
  */};

8. Appium 测试微信公众号demo

AndroidDriver<IWebElement> driver = null;IWebElement element = null;DesiredCapabilities capabilities = new DesiredCapabilities();capabilities.SetCapability("app", "");capabilities.SetCapability("deviceName", "android emulator");capabilities.SetCapability("browserName", "");capabilities.SetCapability("platformName", "android");capabilities.SetCapability("platformVersion", "6.0");//手机操作系统版本capabilities.SetCapability("newCommandTimeout", "300"); ////设置命令超时时间,单位:秒capabilities.SetCapability("unicodeKeyboard", "True");//使用 Unicode 输入法。默认值 falsecapabilities.SetCapability("resetKeyboard", "True"); //在设定了 unicodeKeyboard 关键字的 Unicode 测试结束后,重置输入法到原有状态。capabilities.SetCapability("appPackage", "com.tencent.mm");capabilities.SetCapability("appActivity", ".ui.LauncherUI");capabilities.SetCapability("automationName", "appium");capabilities.SetCapability("fastReset", "false");capabilities.SetCapability("fullReset", "false");capabilities.SetCapability("noReset", "true");DesiredCapabilities option = new DesiredCapabilities();option.SetCapability("androidProcess", "com.tencent.mm:tools");capabilities.SetCapability(ChromeOptions.Capability, option.ToDictionary());Uri serverUri = new Uri("http://127.0.0.1:4723/wd/hub");//System.IO.File.AppendAllText("D:\\PageSources.xml",driver.PageSource);driver = new AndroidDriver<IWebElement>(serverUri, capabilities, TimeSpan.FromSeconds(180));driver.Manage().Timeouts().ImplicitlyWait(TimeSpan.FromSeconds(1));driver.FindElementByXPath("//*[@text='通讯录']").Click();driver.FindElementByXPath("//*[@text='公众号']").Click();driver.FindElementByAccessibilityId("搜索").Click();driver.FindElementByXPath("//*[@text='搜索']").SendKeys("产品测试");driver.FindElementByXPath("//*[@text='产品测试专用']").Click();driver.FindElementByXPath("//*[@text='发现']").Click();driver.FindElementByXPath("//*[@text='微信营业厅']").Click();driver.FindElementByXPath("//*[contains(@text,'微信营业厅')]").Click();Thread.Sleep(6000);((IContextAware)driver).Context = "WEBVIEW_com.tencent.mm:tools"; //切换webviewdriver.FindElementByXPath("//*[text()='登录']").Click();Thread.Sleep(3000);driver.FindElementByXPath("//*[@name='username']").SendKeys("test");  driver.FindElementByXPath("//*[@name='password']").SendKeys("123456"); driver.FindElementByXPath("//button[text()='登录']").Click();

9. Appium 测试手机Chrome浏览器demo

 AndroidDriver<IWebElement> driver = null;IWebElement element = null;DesiredCapabilities capabilities = new DesiredCapabilities();capabilities.SetCapability(MobileCapabilityType.DeviceName, "Android Emulator");capabilities.SetCapability(MobileCapabilityType.BrowserName, MobileBrowserType.Chrome);capabilities.SetCapability(MobileCapabilityType.PlatformName, MobilePlatform.Android);capabilities.SetCapability("platformVersion", "6.0");//手机操作系统版本capabilities.SetCapability("newCommandTimeout", "300"); ////设置命令超时时间,单位:秒。capabilities.SetCapability("unicodeKeyboard", "True");//使用 Unicode 输入法。默认值 falsecapabilities.SetCapability("resetKeyboard", "True"); //在设定了 unicodeKeyboard 关键字的 Unicode 测试结束后,重置输入法到原有状态capabilities.SetCapability("fastReset", "false");capabilities.SetCapability("fullReset", "false");capabilities.SetCapability("noReset", "true");Uri serverUri = new Uri("http://127.0.0.1:4723/wd/hub");driver = new AndroidDriver<IWebElement>(serverUri, capabilities, TimeSpan.FromSeconds(180));driver.Manage().Timeouts().ImplicitlyWait(TimeSpan.FromSeconds(1));WebDriverWait wait = new WebDriverWait(driver, TimeSpan.FromSeconds(10));try{
    ((IContextAware)driver).Context = "CHROMIUM"; //CHROMIUM
    driver.Navigate().GoToUrl("http://12.99.105.46/m/#main");

    //点击登录
    element = wait.Until<IWebElement>((d) =>
    {
        return driver.FindElementByXPath("//*[text()='登录']");
    });

    element.Click();


    //输入用户名
    element = wait.Until<IWebElement>((d) =>
    {
        return driver.FindElementByXPath("//*[@name='username']");
    });
    element.SendKeys("test"); 

    //输入密码
    driver.FindElementByXPath("//*[@name='password']").SendKeys("123456"); 

    driver.FindElementByXPath("//button[text()='登录']").Click();
    Thread.Sleep(3000);}catch (Exception){
    driver.Quit();}

10. 使用Chrome远程调试工具查看浏览器或微信打开的网页控件

可使用谷歌浏览器自带的chrome://inspect/#devices 工具来查看浏览器页面的控件
优点:

  1. 电脑和手机界面双向实时通讯,即可以用PC上的chrome来控制手机浏览器上的页面,手机操作页面时PC Chrome上监控的页面也会试试刷新。
  2. 工具里中Console控制台可以输一些JS命令来控制手机,有些控件我们是需要使用JS去操作的。

工具的使用请参考: 使用Chrome 浏览器调试移动端网页 chrome://inspect/#devices

11. Appium JS操作控件(需要切换到webview模式)

以下方式部分采取angularJS的方式:
JS点击:
对于webview页面中的某些元素,用element.Click()方法操作时会报错,此时我们可以使用JS的方式去点击。

((IContextAware)driver).Context = "WEBVIEW_com.tencent.mm:tools";//切换到webviewelement = driver.FindElementsByXPath("//*[text()='理财产品']").ElementAt(2);IJavaScriptExecutor excutor= (IJavaScriptExecutor)driver;excutor.ExecuteScript(String.Format("arguments[0].{0}", "click()"), element);

文本框输入:
对于webview页面中的某些元素,用element.Click()方法操作时会报错,此时我们可以使用JS的方式去点击。

 //由于很多文本框有JS处理,但有些文本框界面上虽然显示已经输入了,提交时仍会提示为空driver.ExecuteScript("arguments[0].value=arguments[1]", element, "50000"); 
 或driver.ExecuteScript("angular.element(document.getElementsByName('finanAmt')).scope().finanAmt='50000'");

JS操作复选框:

((IContextAware)driver).Context = "WEBVIEW_com.tencent.mm:tools";//切换到webviewIJavaScriptExecutor excutor= (IJavaScriptExecutor)driver;excutor.ExecuteScript(“angular.element(document.getElementsByName('checkboxValue')).scope().checkboxValue=true”);

JS操作下拉框:

 IJavaScriptExecutor js = (IJavaScriptExecutor)driver;
 js.ExecuteScript(“angular.element(document.getElementsByName('finanAccount')).scope().finanAccount = angular.element(document.getElementsByName('finanAccount')).scope().finanAccountList[1]“);
 js.ExecuteScript(“angular.element(document.getElementsByName('finanAccount')).scope().$apply()“);  //同步 Model的变化js.ExecuteScript(“angular.element(document.getElementsByName('finanAccount')).scope().onFinanSelect(angular.element(document.getElementsByName('finanAccount')).scope().finanAccount)“);  //触发控件操作后调用的方法

12. UiAutomator获取webview里面ui元素的方法

Android系统一般自带一个TalkeBack功能(设置-辅助功能-TalkBack)。但这里千万要注意:打开之后整个系统的操作都变得不同了!!滑动界面需要两个手指,单击变成双击。
打开之后用UiAutomatorViewer获取webview界面的内容,你就会发现原来webview里面获取不到的ui元素,现在已经可以可以获取到了。即使现在你关掉TalkBack,也能获取到,除非重启手机才会回到不能获取的状态,所以开启后我们可以立即关闭,以方便操作。


5 Reply   |  Until 2018-8-25 | 984 View

xiaoran
发表于 2018-7-3

666

评论列表

  • 加载数据中...

编写评论内容

egyshell
发表于 2018-7-10

非常感谢


评论列表

  • 加载数据中...

编写评论内容

strafficb
发表于 2018-8-3

666777

评论列表

  • 加载数据中...

编写评论内容

makemoney
发表于 2018-8-25

666

评论列表

  • 加载数据中...

编写评论内容

makemoney
发表于 2018-8-25

666

评论列表

  • 加载数据中...

编写评论内容
LoginCan Publish Content

HYBBS © 2016. All Rights Reserved. 老农民

Powered by HYBBS Version 2.3.2