使用PO+跨平台改造Macaca示例(APP端)
在学习完Macaca基础后,就迫不及待的模仿着Macaca示例项目,开始了测试用例的开发,并且在几天时间里就完成了几个页面的测试。然而,此时项目的所有代码都放在一个.py文件里,该文件已有上千行代码,重复代码很多,维护起来很困难。更大的问题是,我需要给Android和iOS分别写一套代码,这个工作量太多,而且大多是重复代码。
Macaca网络图片为了避免这种情况发生,可以使用PageObject设计模式开发Macaca项目,封装页面对象与用例分离,再把页面对象中定位UI元素的部分脱离,封装成页面UI对象。
项目目录结构
首先,先建立一个下图所示的项目目录结构:
项目目录结构如上图,根目录下面有driver和mail两个文件夹,其中driver是用于存放驱动,mail用于存放项目的测试用例、测试报告以及测试数据等,同目录下还有run_all_test.py文件,用于运行项目的所有自动化用例。
在mail文件夹下,有data、report、test_case三个文件夹,其中data用于存放测试数据,report用于存放测试报告,test_case用于存放测试用例。在test_case文件夹下,有model和page_object两个文件夹,其中model用于存放配置函数及公共类,page_object用于存放页面对象,同目录下还有macaca_case.py这个测试用例文件。
在report文件夹下,有image和template两个文件夹,其中image用于存放测试过程中的截图,template用于存放生成测试报告的工具,例如HTMLTestRunner。
全局配置文件
在项目根目录下新建globalvar.py文件,编写全局配置文件:
global_path = '/Users/hekaiyou/PycharmProjects/po-sample-python/'
server_url = 'http://localhost:3456/wd/hub'
ios_capabilities = {
'platformName': 'ios',
'deviceName': 'iPhone 6s',
'app': 'ios-app-bootstrap.app',
}
android_capabilities = {
'platformName': 'android',
'app': 'android_app_bootstrap-debug.apk',
}
platform = 'android'
在全局配置文件中设置项目的绝对路径、Android与iOS平台的驱动参数、当前测试平台的名称。以后只要更改当前测试平台的名称,即platform参数,就可以选择当前要测试的是Android还是iOS设备。
编写公共模块
在.../mail/test_case/model文件夹下新建driver.py文件,编写公共驱动文件,它可以根据platform参数去调用对应平台的驱动:
from macaca import WebDriver
import globalvar
def drivers():
desired_caps = {}
if globalvar.platform == 'ios':
desired_caps.update(globalvar.ios_capabilities)
elif globalvar.platform == 'android':
desired_caps.update(globalvar.android_capabilities)
else:
pass
desired_caps['app'] = globalvar.global_path + 'driver/' + desired_caps['app']
driver = WebDriver(desired_caps, globalvar.server_url)
return driver
在.../mail/test_case/model文件夹下新建function.py文件,编写公共函数文件,它简单的实现截图函数、切换WebView函数、切换Native函数:
from globalvar import global_path
def insert_img(driver, file_name):
file_path = global_path + '/mail/report/image/' + file_name + '.png'
driver.save_screenshot(file_path)
def switch_to_webview(driver):
contexts = driver.contexts
driver.context = contexts[-1]
return driver
def switch_to_native(driver):
contexts = driver.contexts
driver.context = contexts[0]
return driver
在.../mail/test_case/model文件夹下新建appunit.py文件,编写公共测试类文件,它会调用公共驱动文件来初始化WebDriver服务器:
import unittest
from retrying import retry
from driver import drivers
class AppTest(unittest.TestCase):
@classmethod
def setUpClass(cls):
cls.driver = drivers()
cls.initDriver()
@classmethod
def tearDownClass(cls):
cls.driver.quit()
@classmethod
@retry
def initDriver(cls):
print("正在连接服务器...")
cls.driver.init()
在.../mail/test_case/model文件夹下新建base.py文件,编写基础类文件,它接收页面UI对象中的定位字典,再根据字典内容返回查找到的元素:
from globalvar import platform
class Base(object):
def __init__(self, driver):
self.driver = driver
self.platform = platform
def selector(self, location):
by = None
value = None
if 'by' in location:
by = location['by']
value = location['value']
······
else:
pass
if by == 'xpath':
return self.xpath(value)
······
else:
pass
def xpath(self, xpath):
return self.driver.element_by_xpath(xpath)
······
编写页面UI对象
在.../mail/test_case/page_ui_object文件夹下新建home_page_ui.py文件,编写HOME页面UI对象类文件,它将定位字典传给基础类以获取对应的元素对象:
from base import Base
class HomePageUI(Base):
def home_button_loc(self):
return self.selector({
'ios_by': 'name',
'ios_value': 'HOME',
'android_by': 'wait_name',
'android_value': 'HOME',
})
······
定位字典的结构有两种,当iOS与Android的UI元素的定位方法与参数各不相同时:
{
'ios_by': 'ios_by',
'ios_value': 'ios_value',
'android_by': 'android_by',
'android_value': 'android_value',
}
当iOS与Android的UI元素的定位方法与参数相同时:
{
'by': 'by',
'value': 'value',
}
所有页面UI对象类文件的结构都和上面一样。
编写页面对象
在.../mail/test_case/page_object文件夹下新建home_page.py文件,编写HOME页面对象类文件,它调用对应的页面UI对象类,实现当前页面的所有相关操作:
from home_page_ui import HomePageUI
class HomePage(HomePageUI):
def home_button(self):
self.home_button_loc().click()
def list_button(self):
self.list_button_loc().click()
这样的页面对象类非常简洁,代码清晰明了,所有页面对象类文件的结构都和上面一样。
编写测试用例
在.../mail/test_case文件夹下新建macaca_case.py文件,编写测试用例类文件:
import appunit
from time import sleep
from login_page import LoginPage
class MacacaTest(appunit.AppTest):
def test_01_login(self):
login_po = LoginPage(self.driver)
login_po.username_input('中文+Test+12345678')
login_po.password_input('111111')
sleep(2)
login_po.login_button()
······
执行测试用例
在项目根目录下新建run_all_test.py文件,编写用例执行代码文件:
import unittest
import time
from HTMLTestRunnerCN import HTMLTestRunner
from globalvar import global_path
test_dir = './mail/test_case'
discover = unittest.defaultTestLoader.discover(test_dir, pattern = '*_case.py')
if __name__ == "__main__":
now = time.strftime("%Y-%m-%d %H_%M_%S")
filename = global_path + '/mail/report/'+now+'报告.html'
fp = open(filename, 'wb')
runner = HTMLTestRunner(stream=fp, title='自动化测试报告', description='测试用例描述', tester='测试人员')
runner.run(discover)
fp.close()
这样我们就实现了一个用例对象、页面对象、页面UI对象分离的开发模式,没有重复代码,又易于维护,美滋滋...
为什么选择?有的人喜欢创造世界,他们做了程序员。有的人喜欢拯救世界,他们做了测试员。