转-Appium实战-基类的构建

套用UnitTest构建测试

UnitTest是Python的测试框架,咱们此次的实战就是基于这个测试框架来构建的,因此就把基类用UnitTest构建起来。css

扫盲,什么是UnitTest?怎么用UnitTest?python

  1. 《Hello PyUnitTest》
  2. 《为你的测试结果打印案例名称》
  3. 《测试多个类》
  4. 《测试用例的初始化和结束函数》
  5. 《引入其余类》
  6. 《第一个web自动化》
  7. 《加上测试报告》
  8. 《测试多个文件》

以上是我本身在学习UnitTest过程当中作的一些记录。android

测试案例的颗粒度必需要细,并且测试一个点以后最好进行环境的还原,所以个人基类是这么写的。web

新建一个文件叫BestTestCase,而后输入下面的代码:bash

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,还原环境。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最大的好处了。

最后

最底层的构建就先这样处理了。固然还有其余方法须要构建,好比某个页面的元素定位方法。已经方法的调用。

相关文章
相关标签/搜索