软件测试教程 自动化测试appium篇

软件测试教程 自动化测试appium篇


本课程主要讲解自动化测试工具appium。

下面以android app测试为例,讲解appium的基本使用方法

appium概念

appium安装配置

一个测试样例

Appium 概念

Appium是一个移动端的自动化框架,可用于测试原生应用,移动网页应用和混合型应用,且是跨平台的。可用于IOS和Android以及firefox的操作系统。

Appium使用WebDriver的json wire协议,来驱动Apple系统的UIAutomation库、Android系统的UIAutomator框架。

原生的应用是指用android或ios的sdk编写的应用。原生应用程序看起来(外观)和运行起来(性能)是最佳的。

移动网页应用是指网页应用,HTML5应用程序使用标准的Web技术,通常是HTML5、JavaScript和CSS。

混合应用程序让开发人员可以把HTML5应用程序嵌入到一个细薄的原生容器里面,集原生应用程序和HTML5应用程序的优点(及缺点)于一体。

appium选择了client-server的设计模式。

image

通过上面一张图简单展示了appium的工具原理。

客户端/服务器架构
Appium 的核心是暴露 REST API 的网络服务器。它接受来自客户端的连接,监听命令并在移动设备上执行,答复表示执行结果的 HTTP 响应。客户端/服务器架构实际给予了许多可能性:我们可以使用任何有 http 客户端 API 的语言编写我们的测试代码,不过选一个Appium 客户端程序库 使用更容易。我们可以把服务器放在另一台机器上,而不是执行测试的机器。

会话(session)
自动化始终在一个会话的上下文中执行,这些客户端程序库以各自的方式发起与服务器的会话,但都以发给服务器一个 POST /session 请求结束,请求中包含一个被称作 'desired capabilities' 的 JSON 对象。这时服务器就会开启这个自动化会话,并返回一个用于发送后续命令的会话 ID。

Desired Capabilities
Desired capabilities 是一些发送给 Appium 服务器的键值对集合 (比如 map 或 hash),告诉服务器我们想要启动什么类型的自动化会话。也有各种可以在自动化运行时修改服务器行为的 capabilities。例如,我们可以把 platformName capability 设置为 iOS,告诉 Appium 我们想要 iOS 会话,而不是 Android 或者 Windows 会话。我们也可以设置 safariAllowPopups capability 为 true ,确保我们在 Safari 自动化会话中可以使用 javascript 打开新窗口。有关 Appium capabilities 的完整列表,请参阅 capabilities doc

Appium 服务器
Appium 是用 Node.js 写的服务器。它可以从源码构建安装或者从 NPM 直接安装:

$ npm install -g appium
$ appium

Appium 客户端
有多个客户端程序库(Java、Ruby、Python、PHP,、JavaScript 和 C# 的)支持 Appium 对 WebDriver 协议的扩展,你需要用这些客户端程序库代替通常的 WebDriver 客户端。在这里浏览所有程序库的列表。

Appium.app, Appium.exe
有 Appium 服务器的图形界面包装器可以下载。它们打包了 Appium 服务器运行需要的所有东西,所以你不需要为 Node 而烦恼。它们还提供一个 Inspector 使你可以查看你应用的层级结构,这在写测试时很方便。

appium安装配置

Appium-desktop安装

原来版本的appium-server不再维护,新的工具 是Appium-desktop。

官网地址:http://appium.io/,下载合适的版本

windows平台下载:appium-desktop-setup-1.4.0.exe

双击进行安装,安装过程不需要任何设置

安装完成桌面会生成一个紫色的appium 图标,双击打开。

默认显示监控的 host 和 port ,这和 Appium-Server中是一致的。点击 “Start Server V 1.7.2” 按钮启动服务。

注意:不能在浏览器中设置代理
Traceback (most recent call last):
  File "C:\Users\liujiey\Desktop\test.py", line 20, in setUp

self.driver = webdriver.Remote('http://localhost:4723/wd/hub', desired_caps)

  File "build\bdist.win32\egg\appium\webdriver\webdriver.py", line 36, in __init__

super(WebDriver, self).__init__(command_executor, desired_capabilities, browser_profile, proxy, keep_alive)

  File "C:\Python27\lib\site-packages\selenium-2.48.0-py2.7.egg\selenium\webdriver\remote\webdriver.py", line 87, in __init__

self.start_session(desired_capabilities, browser_profile)

  File "C:\Python27\lib\site-packages\selenium-2.48.0-py2.7.egg\selenium\webdriver\remote\webdriver.py", line 141, in start_session

'desiredCapabilities': desired_capabilities,

  File "C:\Python27\lib\site-packages\selenium-2.48.0-py2.7.egg\selenium\webdriver\remote\webdriver.py", line 199, in execute

response = self.command_executor.execute(driver_command, params)

  File "C:\Python27\lib\site-packages\selenium-2.48.0-py2.7.egg\selenium\webdriver\remote\remote_connection.py", line 395, in execute

return self._request(command_info[0], url, body=data)

  File "C:\Python27\lib\site-packages\selenium-2.48.0-py2.7.egg\selenium\webdriver\remote\remote_connection.py", line 463, in _request

resp = opener.open(request, timeout=self._timeout)

  File "C:\Python27\lib\urllib2.py", line 394, in open

response = self._open(req, data)

  File "C:\Python27\lib\urllib2.py", line 412, in _open

'_open', req)

  File "C:\Python27\lib\urllib2.py", line 372, in _call_chain

result = func(*args)

  File "C:\Python27\lib\urllib2.py", line 1199, in http_open

return self.do_open(httplib.HTTPConnection, req)

  File "C:\Python27\lib\urllib2.py", line 1170, in do_open

r = h.getresponse(buffering=True)
  File "C:\Python27\lib\httplib.py", line 1027, in getresponse

response.begin()

  File "C:\Python27\lib\httplib.py", line 407, in begin

version, status, reason = self._read_status()

  File "C:\Python27\lib\httplib.py", line 371, in _read_status

raise BadStatusLine(line)

BadStatusLine: ''

安装 python-client


python-client 的项目名称叫:Appium-Python-Client。

安装方式:pip install Appium-Python-Client

该项目依赖selenium

在python中测试: from appium import webdriver,导入成功说明安装完成

如果使用其他语言,则安装其他语言的client

安装 JDK

app sdk的运行需要JDK的支持,因此需要先安装JDK

需要配置JAVA_HOME,并设置path路径

安装 Android SDK

进行模拟器测试时,需要SDK的支持。如果真机测试的话,会用到platform-tool,例如adb等工具。因此建议安装SDK。

Android SDK 下载地址:

http://tools.android-studio.org/index.php/sdk

选择对应平台版本下载,这里选择:

installer_r24.4.1-windows.exe

安装SDK:

1、双击进行安装

2、安装过程中会检查JDK

3、安装完毕后会提示运行SDK manager

4、除默认勾选的之外,选择需要下载支持的版本,例如Android 7.0,Android 6.0等,这里选择Android 4.2.2进行下载

设置环境变量:

下面设置环境变量:

“我的电脑” 右键菜单 —> 属性 —> 高级 —> 环境变量 —> 系统变量 —> 新建…

变量名 变量值
ANDROID_HOME D:\android\Android\sdk

找到 path 变量名—> “编辑” 添加:

变量名 变量值
PATH ;%ANDROID_HOME%\platform-tools;%ANDROID_HOME%\tools;

创建模拟器:

1、双击 AVD Manage.exe 启动AVD管理器。

2、点击 “Create…” 按钮,创建Android虚拟机。

3、不要选择超过电脑屏幕分辨率的Device。Target选择SDK下载的版本,Skin选择no skin即可,点击 “OK” 创建完成。

4、在 AVD Manage 工具中选中创建的Android虚拟机,点击 “Start…” 按钮启动。

使用Intel atom模拟时,可能会出现

Starting emulator for AVD 'EMU'
emulator: ERROR: x86 emulation currently requires hardware acceleration!
Please ensure Intel HAXM is properly installed and usable.
CPU acceleration status: HAXM is not installed on this machine

此时可以在sdk manager下载extra中的HAXM或者https://software.intel.com/en-us/articles/intel-hardware-accelerated-execution-manager-intel-haxm下载并安装

异常:未安装sdk
Traceback (most recent call last):
  File "C:\Users\liujiey\Desktop\test.py", line 20, in setUp

self.driver = webdriver.Remote('http://localhost:4723/wd/hub', desired_caps)

  File "build\bdist.win32\egg\appium\webdriver\webdriver.py", line 36, in __init__

super(WebDriver, self).__init__(command_executor, desired_capabilities, browser_profile, proxy, keep_alive)

  File "C:\Python27\lib\site-packages\selenium-2.48.0-py2.7.egg\selenium\webdriver\remote\webdriver.py", line 87, in __init__

self.start_session(desired_capabilities, browser_profile)

  File "C:\Python27\lib\site-packages\selenium-2.48.0-py2.7.egg\selenium\webdriver\remote\webdriver.py", line 141, in start_session

'desiredCapabilities': desired_capabilities,

  File "C:\Python27\lib\site-packages\selenium-2.48.0-py2.7.egg\selenium\webdriver\remote\webdriver.py", line 201, in execute

self.error_handler.check_response(response)

  File "C:\Python27\lib\site-packages\selenium-2.48.0-py2.7.egg\selenium\webdriver\remote\errorhandler.py", line 181, in check_response

raise exception_class(message, screen, stacktrace)

WebDriverException: Message: An unknown server-side error occurred while processing the command. Original error: Could not find adb Please set the ANDROID_HOME environment variable with the Android SDK root directory path.

一个测试样例

创建脚本

一个测试脚本样例:
import os
import unittest
from appium import webdriver
from time import sleep

# Returns abs path relative to this file and not cwd


class ContactsAndroidTests(unittest.TestCase):
    def setUp(self):
        desired_caps = {}
        desired_caps['platformName'] = 'Android'
        desired_caps['platformVersion'] = '4.2'
        desired_caps['deviceName'] = 'Android Emulator'
        //desired_caps['app'] = 'E://ContactManager.apk'
        desired_caps['appPackage'] = 'com.example.android.contactmanager'
        desired_caps['appActivity'] = '.ContactManager'
    
        self.driver = webdriver.Remote('http://localhost:4723/wd/hub', desired_caps)
    
    def tearDown(self):
        self.driver.quit()
    
    def test_add_contacts(self):
        el = self.driver.find_element_by_accessibility_id("Add Contact")
        el.click()
    
        textfields = self.driver.find_elements_by_class_name("android.widget.EditText")
        textfields[0].send_keys("Appium User")
        textfields[2].send_keys("someone@appium.io")
    
        self.assertEqual('Appium User', textfields[0].text)
        self.assertEqual('someone@appium.io', textfields[2].text)
    
        self.driver.find_element_by_accessibility_id("Save").click()
    
        # for some reason "save" breaks things
        alert = self.driver.switch_to_alert()
    
        # no way to handle alerts in Android
        self.driver.find_element_by_android_uiautomator('new UiSelector().clickable(true)').click()
    
        self.driver.press_keycode(3)
        
if __name__ == '__main__':
    suite = unittest.TestLoader().loadTestsFromTestCase(ContactsAndroidTests)
    unittest.TextTestRunner(verbosity=2).run(suite)

Desired Capabilities

desired_caps = {}
desired_caps['platformName'] = 'Android'
desired_caps['platformVersion'] = '4.2'
desired_caps['deviceName'] = 'Android Emulator'
desired_caps['app'] = '../../../sample-code/apps/ContactManager/ContactManager.apk'
desired_caps['appPackage'] = 'com.example.android.contactmanager'
desired_caps['appActivity'] = '.ContactManager'
driver = webdriver.Remote('http://localhost:4723/wd/hub', desired_caps)
  • deviceName:启动哪种设备,是真机还是模拟器?iPhone Simulator,iPad Simulator,iPhoneRetina 4-inch,Android Emulator,Galaxy S4…

    deviceName通过adb devices查看
    
  • platformName:使用哪种移动平台。iOS, Android, orFirefoxOS。

  • platformVersion:指定平台的系统版本。例如指的Android平台,版本为5.1。

  • app:apk文件的路径,appium会尝试先安装该app。

  • appActivity:待测试的app的Activity名字。比如MainActivity、.Settings。注意,原生app的话要在activity前加个”.“。

  • appPackage:待测试的app的Java package。比如com.example.android.myApp, com.android.settings。

appActivity和appPackage一般由开发人员提供,如果无法获取的话,可以通过以下方式获取

1、用winrar打开apk

2、选择AndroidManifest.xml,右键查看文件

3、查看

其中:m a n i f e s t " c o m . e x a m p l e . a n d r o i d . c o n t a c t m a n a g e r"就是appPackage

其中:a c t i v i t y � . C o n t a c t M a n a g e r 就是appActivity

v e r s i o n C o d e   
 v e r s i o n N a m e   
 m i n S d k V e r s i o n   � t a r g e t S d k V e r s i o n   � n a m e   � l a b e l   � i c o n   
 d e b u g g a b l e   � a n d r o i d   * h t t p : / / s c h e m a s . a n d r o i d . c o m / a p k / r e s / a n d r o i d       � p a c k a g e    m a n i f e s t   " c o m . e x a m p l e . a n d r o i d . c o n t a c t m a n a g e r   � 1 . 0    u s e s - s d k   � u s e s - p e r m i s s i o n   � a n d r o i d . p e r m i s s i o n . G E T _ A C C O U N T S     a n d r o i d . p e r m i s s i o n . R E A D _ C O N T A C T S   ! a n d r o i d . p e r m i s s i o n . W R I T E _ C O N T A C T S   
 a p p l i c a t i o n    a c t i v i t y   � . C o n t a c t M a n a g e r   
 i n t e n t - f i l t e r   � a c t i o n   � a n d r o i d . i n t e n t . a c t i o n . M A I N    c a t e g o r y     a n d r o i d . i n t e n t . c a t e g o r y . L A U N C H E R   

appium 定位控件

这里通过appium自带的Inspector工具可以查看对象各种属性。也可以用android sdk自带的uiautomatorviewer.bat来进行定位

1、启动appium server

2、启动模拟器,并安装待测试的apk:ContactManager.apk(将文件拖入模拟器即可)

3、模拟器中打开ContactManager.apk,放在待测试的页面

4、点击Start Inspector session图标

5、在name和value处分别输入以下内容,点击start session启动

appium-1.PNG
platformName  Android
platformVersion  4.2
deviceName  Android Emulator
appPackage  com.example.android.contactmanager
appActivity  ContactManager

界面如下

appium-1png.png

录制脚本

Inspector提供了简单的脚本操作

1、选择录制脚本

2、在页面上做一些操作,比如点击等

3、可以看到页面上有脚本出现,复制脚本即可

定位元素信息

选中一个元素就可以实现定位,页面右侧展示了元素具体的内容

注意:出现不能点击鼠标来进行定位的话,可以选用uiautomatorviewer.bat来进行定位,或者两个结合使用。

appium-2.PNG

id 定位

如果目标设备的API Level低于18则UIAutomatorViewer不能获得对应的Resource ID,只有等于大于18的时候才能使用。

image

resource-id 就是我们理解的id属性了。

使用方法:

driver.findElement(By.id("com.android.calculator2:id/formula"))

name 定位

text就是我们要查找的name!

使用方法:

driver.findElement(By.name("21"))

Accessibility ID定位

这个方法属于Appium扩展的定位方法。

其实,我们的核心是要找到元素的contentDescription属性。它就是元素的 content-desc 。

使用方法:

driver.findElementByAccessibilityId("Add contact").click();

android uiautomator定位

这个方法也属于 Appium(Android)扩展的定位方法。

也就是说一个元素的任意属性都可以通过android uiautomator方法来进行定位,但要保证这种定位方式的唯一性。

使用方法:


driver.find_element_by_android_uiautomator('new UiSelector().text("Custom View")').click()       driver.find_element_by_android_uiautomator('new UiSelector().textContains("View")').click()     
driver.findElementByAndroidUIAutomator('new UiSelector().description("equals")").click();

需要注意的是 description() 方法用的是content-desc属性。

find_elements_by_class_name方法

方法:

  • find_elements_by_class_name()

通过元素class name属性定位所有含有该属性的元素

driver.find_elements_by_class_name('name')

appium API

sendKeys()方法

方法:

  • sendKeys()

用法:

driver.findElements(By.name("Name")).sendKeys("jack");

pressKeyCode()方法

除此之外,appium扩展提供了pressKeyCode()方法。该方法Android特有。

方法:

  • pressKeyCode()

发送一个键码的操作。需要一个入参。

driver.pressKeyCode(29); // 字母“a”
driver.pressKeyCode(3);//KEYCODE_HOME 

switch_to_alert()方法

方法:

  • switch_to_alert()

切换到alert窗口

driver.switch_to_alert()

click()方法

方法:

  • click()

点击某个控件

element.click()

API文档参考:http://blog.csdn.net/yangyinsong815/article/details/51235708

至此,脚本中所有的知识点讲解完毕,现在开始运行它吧!运行之前请确保模拟器打开!

真机运行

模拟器并不能模拟真实的手机执行,最终我们的案例还是需要在真机中进行。

真机和模拟器执行的区别在于:

真机运行需要打开USB调试所有项

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,588评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,456评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,146评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,387评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,481评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,510评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,522评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,296评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,745评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,039评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,202评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,901评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,538评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,165评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,415评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,081评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,085评论 2 352

推荐阅读更多精彩内容