测试铁晔 发表于 2018-5-21 15:53:07

Appium脚本和测试报告(Python)

我知道自学Appium和Python还是一个比较费时间的过程。所以,希望我的学习经历能带刚入坑的小伙伴
少走一些弯路,尽快能用上这么牛逼的一个工具,足矣!

如何用Python语言写Appium测试脚本?我之前写过就是利用AppiumDesktop录制脚本,但是这个录制出
来的脚本可读性比较差,今天我所讲到的就是修改录制出来的脚本,再模仿这个脚本去编写其他的测试
用例。

今天讲到的测试App是我们公司正在开发的一款,使用的是iOS模拟器(当然真机也可以,配置一下真机
的信息就好了,因为我们公司没有iPhone测试机,需要使用个人的手机进行测试,对于手机就比较爱惜
了,主要还是因为穷,换个手机太难了。所以我就等在模拟器上验证以后再在真机上实践),实现的功
能为登录。

以下为录制出来的脚本:

# This sample code uses the Appium python client
# pip install Appium-Python-Client
# Then you can paste this into a file and simply run with Python
from appium import webdriver

#配置模拟器信息
caps = {}
caps["platformName"] = "ios"
caps["deviceName"] = "iPhone 7 Plus"
caps["app"] = "/Users/guxuecheng/Desktop/SmartLife.app"
caps["platformVersion"] = "10.3"
caps["moReset"] = True
driver = webdriver.remote("http://0.0.0.0:4723/wd/hub", caps)

/#滑动屏幕(第一次安装App的时候会有引导页)
TouchAction(driver)
.press({x: 335, y: 365})
.moveTo({x: -209: y: 2})
.release()

TouchAction(driver)
.press({x: 323, y: 387})
.moveTo({x: -213: y: 21})
.release()

#点击“立即使用”按钮进入登录页面
TouchAction(driver).tap([(212, 635)])
/#点击输入手机号textfield(这里是用xpath进行定位,但其实这种方法定位很不好,官方并不推荐,因为
能否定位成功需要看人品,如果没有accessibility_id,最好还是和开发说一下,让万能的开发工程师在代码里
给加上吧,我在后面修改好的脚本就是用的accessibility_id进行定位的)
el1 = driver.find_element_by_xpath("//XCUIElementTypeApplication[@name=\"i+智慧社区 \"]/XCUIEleme
ntTypeWindow/XCUIElementTypeOther/XCUIElementTypeOther/XCUIElementTypeOther
/XCUIElementTypeOther/XCUIElementTypeOther/XCUIElementTypeImage/XCUIElementTypeOther/XCU
IElementTypeOther/XCUIElementTypeOther/XCUIElementTypeTextField")

el1.click()
el1.send_keys("12212345678")

#点击输入密码textfield
el2 = driver.find_element_by_xpath("//XCUIElementTypeApplication[@name="i+智慧社区"]/XCUIElemen
tTypeWindow/XCUIElementTypeOther/XCUIElementTypeOther/XCUIElementTypeOt
her/XCUIElementTypeOther/XCUIElementTypeOther/XCUIElementTypeImage/XCUIElementTypeOther
/XCUIElementTypeOther/XCUIElementTypeOther/XCUIElementTypeSecureTextField")

el2.click()
el2.send_keys("6666666")

/#点击登录按钮
TouchAction(driver).tap([(217, 377)])

/#登录以后立即退出(网络稍有延迟,你都看不到首页就被退出了,所以这里我们会设置等待时间)
driver.quit()
看着上面这个脚本感觉咋样?你们有看着舒服的你们就这么搞,反正我看着不够清晰,然后就想办法对
这个代码进行修改。

修改后的代码如下:

# This sample code uses the Appium python client
# pip install Appium-Python-Client
# Then you can paste this into a file and simply run with Python
# -*- coding: utf-8 -*-
#!/usr/bin/env python3
from appium import webdriver
import time
time.sleep(3)

caps = {}
caps["platformName"] = "ios"
caps["deviceName"] = "iPhone 7 Plus"
caps["app"] = "/Users/guxuecheng/Desktop/SmartLife.app"
caps["platformVersion"] = "10.3"
caps["moReset"] = True

driver = webdriver.Remote("http://0.0.0.0:4723/wd/hub", caps)

#滑动屏幕
driver.swipe ( 356, 598, -313, -6,3)
#滑动屏幕暂停1s,确保滑动到下一页
time.sleep(1)
driver.swipe ( 356, 598, -313, -6,3)
time.sleep(1)

#立即体验---进入登录页面
driver.tap([(212,635)])
time.sleep(1)

#注意这里使用的是accessibility_id进行定位
el1 = driver.find_element_by_accessibility_id("_phoneNumT")
el1.click()
el1.send_keys("12212345678")

el2 = driver.find_element_by_accessibility_id("_passwordT")
el2.click()
el2.send_keys("6666666")

#登录
driver.find_element_by_accessibility_id("_loginButton").click()

time.sleep(10)

driver.quit()
我个人还是觉得修改后的代码稍微美观一些。以上脚本所用到的接口都是Appium Python Client
如果你确实是一个小白(高手请绕道),在这里请思考一个问题:这样一个脚本在实际工作中能用吗?
能满足测试要求吗?能替代手工吗?如果以上答案都为否定,那么该如何改进呢?
可爱的Python给我们提供了一个很好的框架 unittest.
利用这个框架我们就能写出更加完善的测试用例了。

下面就是用了unittest框架实现了登录功能的脚本

# This sample code uses the Appium python client
# pip install Appium-Python-Client
# Then you can paste this into a file and simply run with Python
# -*- coding: utf-8 -*-
#!/usr/bin/env python3
from appium import webdriver
#from time import sleep
import time
#from unittest import TestCase
import unittest
#import sys
#import os
time.sleep(3)

caps = {}
caps["platformName"] = "ios"
caps["deviceName"] = "iPhone 7 Plus"
caps["app"] = "/Users/guxuecheng/Desktop/SmartLife.app"
caps["platformVersion"] = "10.3"
caps["moReset"] = True

#TestCase类,所有测试用例类继承的基本类
class LoginTest(unittest.TestCase):

    #setUp()方法用于测试用例执行前的初始化工作。如测试用例中需要访问数据库,可以在setUp中
建立数据库链接
    #并进行初始化。如测试用例需要启动Appium服务,则需要在该方法内启动Appium服务。
    def setUp(self):
      self.driver = webdriver.Remote("http://0.0.0.0:4723/wd/hub", caps)

    #tearDown()方法用于测试用例执行之后的善后工作。如关闭数据库连接,退出应用。
    #无论这个方法写在哪里,都是最后才执行
    def tearDown(self):
      self.driver.quit()

    #具体的测试用例,必须要以test开头
    def test_start(self):
      self.driver.swipe ( 356, 598, -313, -6,3)
      time.sleep(1)
      self.driver.swipe ( 356, 598, -313, -6,3)
      time.sleep(1)

      #立即体验---进入登录页面
      self.driver.tap([(212,635)])
      time.sleep(1)

      el1 = self.driver.find_element_by_accessibility_id("_phoneNumT")
      el1.click()
      el1.send_keys("13012345678")

      el2 = self.driver.find_element_by_accessibility_id("_passwordT")
      el2.click()
      el2.send_keys("6666666")

      #点击登录按钮
    self.driver.find_element_by_accessibility_id("_loginButton").click()
      time.sleep(3)
      self.assertIsNotNone(self.driver.find_element_by_accessibility_id('tab_button_home_normal'),['进
入到首页,登录成功,TRUE'])
      time.sleep(10)

if __name__ == '__main__':

    #构造测试集defaultTestLoader()即TestLoader()测试用例加载器,包括多个加载测试用例的
方法,返回一个测试套件
    #loadTestsFromTestCase()根据给定的测试类,获取其中的所有测试方法,并返回一个测试套件
    suite = unittest.TestLoader().loadTestsFromTestCase(LoginTest)



测试铁晔 发表于 2018-5-21 15:53:56

    #unittest框架的TextTestRunner()类,通过该类下面的run()方法来运行suite所组装的测试用例,
入参为suite测试套件
    unittest.TextTestRunner(verbosity=2).run(suite)

    #上面两行代码可以换成下面一行
    #unittest.main()
这段代码用了比较多的注释,所以看起来比较凌乱,主要是想让和我一样的小白尽快学一些东西,如果
还有看不懂的可以留言讨论,互相学习。

再附加一个注册功能的脚本

这个脚本就是模仿登录脚本写的,主要是为接下来的内容做准备

# -*- coding: utf-8 -*-
#!/usr/bin/env python3

from appium import webdriver
import timeimport unittest

caps = {}
caps["platformName"] = "ios"
caps["deviceName"] = "iPhone 7 Plus"
caps["app"] = "/Users/guxuecheng/Desktop/SmartLife.app"
caps["platformVersion"] = "10.3"
caps["moReset"] = True

#TestCase类,所有测试用例类继承的基本类
class RegistTest(unittest.TestCase):

def setUp(self):
self.driver = webdriver.Remote("http://0.0.0.0:4723/wd/hub", caps)

def tearDown(self):
self.driver.quit()
#具体的测试用例,必须要以test开头
def test_start(self):
#滑动屏幕
self.driver.swipe ( 356, 598, -313, -6,3)
time.sleep(1)
self.driver.swipe ( 356, 598, -313, -6,3)
time.sleep(1)

#立即体验---进入登录页面
el0 = self.driver.find_element_by_accessibility_id("useAppRightNow")
el0.click()
#点击注册按钮
el1 = self.driver.find_element_by_accessibility_id("_registerButton")
el1.click()
#输入需要注册手机号
el2 = self.driver.find_element_by_accessibility_id("_registerPhoneNumT")
el2.click()
el2.send_keys("12212345678")
#输入验证码
el3 = self.driver.find_element_by_accessibility_id("_registerSecurityT")
el3.click()
el3.send_keys("这里使用的是万能验证码")
#输入密码
el4 = self.driver.find_element_by_accessibility_id("_registerPasswordT")
el4.click()
el4.send_keys("6666666")
#点击同意服务协议
el5 = self.driver.find_element_by_accessibility_id("changeSwitch")
el5.click()
#点击注册按钮
el6 = self.driver.find_element_by_accessibility_id("_registerbutton")
el6.click()
time.sleep(3)

self.assertIsNotNone(self.driver.find_element_by_accessibility_id('住房信息'),['注册成功,TRUE'])
time.sleep(10)

if __name__ == '__main__':

suite = unittest.TestLoader().loadTestsFromTestCase(RegistTest)

unittest.TextTestRunner(verbosity=2).run(suite)

self.driver.quit()
上面已经给出两段脚本,我们如果分两次执行也没有问题,但是如果脚本达到200个呢?还这样一个个
执行吗?显然是不行的。友善的Python也为我们解决了这样一个问题。她给我们提供了这个便利的同时
也制定了一些规则,我们必须遵守这些规则才行。

合并脚本测试

规则:上面我们已经写了两个脚本,对于这两个py文件在命名上我们并没有什么限制,但是如果想要合
并起来一并执行那就在命名上需要遵守规则了。这些文件的命名必须要以test开头。然后我们在新建一
个py文件(这个文件命名就没有那个限制了),这些文件需要放到一个文件夹里才行,对于这个文件(
这个文件我们暂且命名为test_suite.py)需要写的内容很少,代码如下:

import unittest

#test_register是文件名,RegistTest是该文件里定义的一个类
from test_register import RegistTest
from test_login import LoginTest

if __name__ == '__main__':
    suite = unittest.TestSuite()
    suite.addTests(unittest.TestLoader().loadTestsFromTestCase(RegistTest))
    suite.addTests(unittest.TestLoader().loadTestsFromTestCase(LoginTest))

#这一步是在当前文件夹里自动生成一个测试报告,测试报告名称就叫:UnittestTextReport.txt.
   with open('./UnittestTextReport.txt', 'a') as f:
      runner = unittest.TextTestRunner(stream=f, verbosity=2)
       runner.run(suite)

大家看一下这个测试报告,不是很直观:



但是这个测试报告又有他的好处,对于错误的用例,他会将错误的地方标注出来:


这样呢也方便我们排查。就是有一点,不利于统计。下面贴一个更生动的测试报告截图,也是这两个
用例的执行结果:



这个测试报告怎么样?
测试报告生成的时间,多少用例执行通过,多少用例失败,一共有多少用例被执行,都一目了然。
这样一个直观的测试报告就是利用HTMLTestRunner.py这个文件来实现的,首先你需要先下载这样一个
文件放在脚本所在的同一个文件夹里。然后修改你的test_suite.py文件,修改以后如下:
import unittest

from test_register import RegistTest
from test_login import LoginTest

import HTMLTestRunner


if __name__ == '__main__':
suite = unittest.TestSuite()
suite.addTests(unittest.TestLoader().loadTestsFromTestCase(RegistTest))
suite.addTests(unittest.TestLoader().loadTestsFromTestCase(LoginTest))


with open('HTMLReport.html', 'wb') as f:
runner = HTMLTestRunner.HTMLTestRunner(stream=f,
title='MathFunc Test Report',
description='generated by HTMLTestRunner.',
verbosity=2
)
runner.run(suite)

然后再执行修改后的脚本,就会在脚本所在的文件夹里自动生成一个html格式的测试报告,打开该报
告就是如上所示。
你学会了吗?如果还有疑问欢迎留言讨论,互相学习,共同进步!

部分脚本的补充(解释):

1、 一个python的文件有两种使用的方法,第一是直接作为脚本执行,第二是import到其他的python
脚本中被调用(模块重用)执行。因此if name == 'main': 的作用就是控制这两种情况执行代码的过程
,在if name == 'main': 下的代码只有在第一种情况下(即文件作为脚本直接执行)才会被执行,而im
port到其他脚本中是不会被执行的。

2、Desired Capabilities:
De



页: [1]
查看完整版本: Appium脚本和测试报告(Python)