套用UnitTest构建测试
UnitTest是Python的测试框架,我们这次的实战就是基于这个测试框架来构建的,所以就把基类用UnitTest构建起来。
扫盲,什么是UnitTest?怎么用UnitTest?
以上是我自己在学习UnitTest过程中做的一些记录。
测试案例的颗粒度必须要细,而且测试一个点之后最好进行环境的还原,因此我的基类是这么写的。
新建一个文件叫BestTestCase,然后输入下面的代码:
import unittest
from appium import webdriver
import config
class AppTestCase(unittest.TestCase):
def setUp(self):
desired_caps = {
'platformName': config.CONNECT['platformName'],
'platformVersion': config.CONNECT['platformVersion'],
'deviceName': config.CONNECT['deviceName'],
'appPackage': config.CONNECT['appPackage'],
'appActivity': config.CONNECT['appActivity']
}
self.driver = webdriver.Remote(config.CONNECT['baseUrl'], desired_caps)
def tearDown(self):
self.driver.quit()
这样写,只要其他的测试类来集成这个类就行了,就不用再每个测试类中写setUp和tearDown了,而且每次测试完功能点后就会自动退出APP,还原环境。
导入包有一个import config,这个是我做的一个配置文件,命名为config.py,我把一些不常修改的参数往这里丢。比如这个初始化的参数,我就这样放了。
config.py
CONNECT = {
'platformName': 'Android',
'platformVersion': '4.4.4',
'deviceName': '5136b01e',
'appPackage': 'com.weizq',
'appActivity': 'com.zztzt.android.simple.app.MainActivity',
"baseUrl": "http://127.0.0.1:4723/wd/hub"
}
构建通用方法
UnitTest的基类构建完之后,我们要对基础的方法做一些封装。
常用的功能大概有这些:寻找元素的各种方法,系统的按键,滑动,获取坐标,截图等。废话不多说,先上代码:
from selenium.webdriver.support.expected_conditions import NoSuchElementException
from selenium.webdriver.support.ui import WebDriverWait
import time as t
import config
class WebDdriver(object):
def __init__(self, driver):
self.driver = driver
def __str__(self):
return 'webDdriver'
def find_element(self, *loc):
"""
定位元素,定位正确后返回元素的信息,外部调用传入元组参数必须有*,
例如:
find_element(*self.native_caixun)
:param loc: 元组类型,结构必须是(By.NAME, u'财讯')
:return: element
"""
try:
element = WebDriverWait(self.driver, 10).until(lambda x: x.find_element(*loc))
return element
except NoSuchElementException, e:
print 'Error details :%s' % (e.args[0])
def find_elements(self, *loc):
"""
定位元素,定位正确后返回元素的信息,外部调用传入元组参数必须有*,
例如:
find_elements(*self.native_caixun)
:param loc: 元组类型,结构必须是(By.NAME, u'财讯')
:return: elements
"""
try:
# return self.driver.find_elements(*loc)
elements = WebDriverWait(self.driver, 10).until(lambda x: x.find_elements(*loc))
return elements
except NoSuchElementException, e:
print 'Error details :%s' % (e.args[0])
def get_title(self):
"""
获取页面的标题
:return: str, 页面的标题
"""
title = self.find_elements(*config.COMMON['view_title'])[0].text
return title
@property
def wait(self):
t.sleep(5)
def getScreenshot(self, name, url, form='png'):
t.sleep(2)
self.driver.get_screenshot_as_file(url + name + "." + form)
def sysback(self):
"""
系统的返回按钮
:return: None
"""
self.driver.keyevent(4)
def get_size(self):
"""
获取当前屏幕的分辨率
:return: int, x*y
"""
size = self.driver.get_window_size()
return size
def swipe_to_up(self):
"""
从下往上滑动
:return: None
"""
window_size = self.get_size()
width = window_size.get("width")
height = window_size.get("height")
self.driver.swipe(width / 2, height * 3 / 4, width / 2, height / 4, 500)
def swipe_to_down(self):
"""
从上往下滑动
:return: None
"""
window_size = self.get_size()
width = window_size.get("width")
height = window_size.get("height")
self.driver.swipe(width / 2, height / 4, width / 2, height * 3 / 4, 500)
def swipe_to_left(self):
"""
从右往左滑动
:return: None
"""
window_size = self.get_size()
width = window_size.get("width")
height = window_size.get("height")
self.driver.swipe(width / 4, height / 2, width * 3 / 4, height / 2, 500)
def swipe_to_right(self):
"""
从左往右滑动
:return: None
"""
window_size = self.get_size()
width = window_size.get("width")
height = window_size.get("height")
self.driver.swipe(width * 4 / 5, height / 2, width / 5, height / 2, 500)
def getLocation(self, *loc):
"""
获取元素的定位信息,外部调用传入元组参数必须有*,
例如:
(*self.native_caixun)
:param loc: 元素的定位方式
:return: list, [x, y]
"""
locaX = self.find_element(*loc).location.get('x')
locaY = self.find_element(*loc).location.get('y')
rst = [locaX, locaY]
return rst
注释我应该写的很清楚了,就把一些比较重要的地方做一些解释,在init中先把刚刚构建的driver初始化了,让类中的其他方法可以使用driver。
两个最重要的方法,find_element和find_elements做一下重点说明。正常来说,找元素的方法有这么几种,name、id、xpath、tag、class_name、link_text等,但是如果做这么多封装,代码就会显得非常多,当然,如果没有其他办法的话,那就只能老老实实的枚举了,但是如果去看webdriver的源码,就能找到这么一个方法。源码是这样的:
class By(object):
"""
Set of supported locator strategies.
"""
ID = "id"
XPATH = "xpath"
LINK_TEXT = "link text"
PARTIAL_LINK_TEXT = "partial link text"
NAME = "name"
TAG_NAME = "tag name"
CLASS_NAME = "class name"
CSS_SELECTOR = "css selector"
@classmethod
def is_valid(cls, by):
for attr in dir(cls):
if by == getattr(cls, attr):
return True
return False
那么也就是说,在webdriver的方法源码中已经帮我们枚举了,那么我们就可以直接调用这个方法,,做参数传入就行了。方法有做入参检查,因此传入参数要求是tuple的类型,所以元素我就全部剥离出来用配置文件处理,配置文件的内容我们就这样写:
from selenium.webdriver.common.by import By
CAIXUN = {
'tuijian': (By.NAME, u'推荐'),
'gupiao': (By.NAME, u'股票'),
'jijin': (By.NAME, u'基金'),
'zhaiquan': (By.NAME, u'债券'),
'xinsanban': (By.NAME, u'新三板'),
'zixunneirong': (By.ID, 'com.weizq:id/new_title'), # 财讯记录列表,用find_elements调用
'imgurl': '/Users/SvenWeng/PycharmProjects/WeStock/img/caixun/',
'title': [
['tuijian', 'tuijianneirong'],
['gupiao', 'gupiaoneirong'],
['jijin', 'jijinneirong'],
['zhaiquan', 'zhaiquanneirong'],
['xinsanban', 'xinsanbanneirong']
],
'jiahao': (By.CLASS_NAME, 'android.widget.ImageView'),
'pindao': (By.ID, 'com.weizq:id/text_item'),
}
中间的那些元素就直接这样配置,如果开发进行了一些修改,那么我们就直接改这里就行了。这就是Page Object最大的好处了。
最后
最底层的构建就先这样处理了。当然还有其他方法需要构建,比如某个页面的元素定位方法。已经方法的调用。