美文网首页
使用PO+跨平台改造Macaca示例(APP端)

使用PO+跨平台改造Macaca示例(APP端)

作者: 何小有 | 来源:发表于2018-01-12 18:35 被阅读205次

    在学习完Macaca基础后,就迫不及待的模仿着Macaca示例项目,开始了测试用例的开发,并且在几天时间里就完成了几个页面的测试。然而,此时项目的所有代码都放在一个.py文件里,该文件已有上千行代码,重复代码很多,维护起来很困难。更大的问题是,我需要给Android和iOS分别写一套代码,这个工作量太多,而且大多是重复代码。

    Macaca网络图片

    为了避免这种情况发生,可以使用PageObject设计模式开发Macaca项目,封装页面对象与用例分离,再把页面对象中定位UI元素的部分脱离,封装成页面UI对象。

    项目GitHub——传送门

    项目目录结构

    首先,先建立一个下图所示的项目目录结构:

    项目目录结构

    如上图,根目录下面有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对象分离的开发模式,没有重复代码,又易于维护,美滋滋...

    为什么选择?有的人喜欢创造世界,他们做了程序员。有的人喜欢拯救世界,他们做了测试员。

    相关文章

      网友评论

          本文标题:使用PO+跨平台改造Macaca示例(APP端)

          本文链接:https://www.haomeiwen.com/subject/rjpfoxtx.html