• 软件测试开发实战 | 记录写装饰器时踩的几个坑


    本文转载自霍格沃兹测试学院优秀学员indeyo 小玲儿的学习笔记,原文链接:http://qrcode.testing-studio.com/f?from=bokeyuan&url=https://ceshiren.com/tag/精华帖

    背景

    装饰器是python里面一个很有用的语法糖( Syntactic Sugar),可以减少大量重复代码的编写。

    装饰器学习地址:https://realpython.com/primer-on-python-decorators/ 34

    刚好最近学习了app自动化框架的异常处理,存在一定重复代码,准备当作题材,拿来练习一下装饰器。

    下面记录一下装饰器的踩坑之路。

    坑 1:Hint: make sure your test modules/packages have valid Python names.

    报错信息

    test_market.py:None (test_market.py)
    ImportError while importing test module 'D:projectHogwarts_11	est_appium	estcase	est_market.py'.
    Hint: make sure your test modules/packages have valid Python names.
    Traceback:
    test_market.py:9: in <module>
        from test_appium.page.app import App
    ..pageapp.py:12: in <module>
        from test_appium.page.base_page import BasePage
    ..pagease_page.py:16: in <module>
        from test_appium.utils.exception import exception_handle
    ..utilsexception.py:11: in <module>
        from test_appium.page.base_page import BasePage
    E   ImportError: cannot import name 'BasePage' from 'test_appium.page.base_page' (D:projectHogwarts_11	est_appiumpagease_page.py)
    

    原因
    exception.py 文件和 base_page.py 文件之间存在相互调用关系。

    解决方案
    把循环调用的包引入信息放在函数内。只要一方的引用信息放在函数里即可,不必两边都放。

    我只在 exception.py 文件里改了,base_page.py 保持不变。

    exception.py

    def exception_handle(func):
        def magic(*args, **kwargs):
            # 防止循环调用报错
            from test_appium.page.base_page import BasePage
            # 获取BasePage实例对象的参数self,这样可以复用driver
            _self: BasePage = args[0]
    ...
    

    坑 2:IndexError: tuple index out of range

    报错信息

    test_search.py:None (test_search.py)
    test_search.py:11: in <module>
        from test_appium.page.app import App
    ..pageapp.py:12: in <module>
        from test_appium.page.base_page import BasePage
    ..pagease_page.py:52: in <module>
        class BasePage:
    ..pagease_page.py:74: in BasePage
        def find(self, locator, key=None):
    ..pagease_page.py:50: in exception_handle
        return magic()
    ..pagease_page.py:24: in magic
        _self: BasePage = args[0]
    E   IndexError: tuple index out of range
    

    原因
    第一次写装饰器真的很容易犯这个错,来看下哪里写错了

    def decorator(func):
        def magic(*args, **kwargs):
            _self: BasePage = args[0]
            ...
            return magic(*args, **kwargs)
        # 这里的问题!!!不应该返回函数调用,要返回函数名称!!!
        return magic()
    

    为什么返回函数调用会报这个错呢?

    因为调用magic()函数的时候,没有传参进去,但是magic()里面引用了入参,这时args没有值,自然就取不到args[0]了。

    解决方案
    去掉括弧就好了

    def decorator(func):
        def magic(*args, **kwargs):
            _self: BasePage = args[0]
            ...
            return magic(*args, **kwargs)
        # 返回函数名,即函数本身
        return magic
    

    坑 3:异常处理只执行了1次,自动化无法继续

    报错信息
    主要是定位元素过程中出现的各种异常,NoSuchElementExceptionTimeoutException等常见问题。

    原因
    异常处理后,递归逻辑写得不对。return func()执行了func(),跳出了异常处理逻辑,所以异常处理只执行一次。

    正确的写法是 return magic()

    感觉又是装饰器小白容易犯的错误…emmm…

    解决方案
    为了直观,已过滤不重要代码,异常处理逻辑代码会在文末放出。

    def exception_handle(func):
        def magic(*args, **kwargs):
            _self: BasePage = args[0]
            try:
                return func(*args, **kwargs)
            # 弹窗等异常处理逻辑
            except Exception as e:
                for element in _self._black_list:
                    elements = _self._driver.find_elements(*element)
                    if len(elements) > 0:
                        elements[0].click()
                        # 异常处理结束,递归继续查找元素 
                        # 这里之前写成了return func(*args, **kwargs),所以异常只执行一次!!!!!
                        return magic(*args, **kwargs)
                raise e
        return magic
    

    坑 4:如何复用driver?

    问题
    自己刚开始尝试写装饰器的时候,发现一个问题。

    装饰器内需要用到 find_elements,这时候 driver 哪里来?还有 BasePage 的私有变量 error_max 和 error_count 怎么获取到呢?创建一个 BasePage 对象?然后通过 func 函数来传递 driver ?

    func的driver是私有的,不能外部调用(事实证明可以emmm…)。

    我尝试把异常相关的变量做成公共的,没用,还是无法解决find_elements的调用问题。

    解决方案
    思寒的做法是,在装饰器里面创建一个self变量,取args[0],即函数func的第一个入参self。

    _self: BasePage = args[0]这一简单的语句成功解答了我所有的疑问。

    类函数定义里面 self 代表类自身,因此可以获取 ._driver 属性,从而调用 find_elements。

    坑 5:AttributeError

    找到元素后,准备点击的时候报错

    报错信息

    EINFO:root:('id', 'tv_search')
    INFO:root:None
    INFO:root:('id', 'image_cancel')
    INFO:root:('id', 'tv_agree')
    INFO:root:('id', 'tv_search')
    INFO:root:None
    
    test setup failed
    self = <test_appium.testcase.test_search.TestSearch object at 0x0000018946B70940>
    
        def setup(self):
    >       self.page = App().start().main().goto_search()
    
    test_search.py:16: 
    _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _
    
    self = <test_appium.page.main.MainPage object at 0x0000018946B70780>
    
        def goto_search(self):
    >       self.find(self._search_locator).click()
    E       AttributeError: 'NoneType' object has no attribute 'click'
    
    ..pagemain.py:20: AttributeError
    

    原因
    看了下 find 函数,找到元素后,有返回元素本身

    @exception_handle
        def find(self, locator, key=None):
            logging.info(locator)
            logging.info(key)
            # 定位符支持元组格式和两个参数格式
            locator = locator if isinstance(locator, tuple) else (locator, key)
            WebDriverWait(self._driver, 10).until(expected_conditions.visibility_of_element_located(locator))
            element = self._driver.find_element(*locator)
            return element
    

    那就是装饰器写得不对了

    def exception_handle(func):
        def magic(*args, **kwargs):
            _self: BasePage = args[0]
            try:
                # 这里只是执行了函数,但是没有return
                func(*args, **kwargs)
            # 弹窗等异常处理逻辑
            except Exception as e:
                raise e
        return magic
    

    解决方案
    要在装饰器里面返回函数调用,要不然函数本身的返回会被装饰器吃掉。

    def exception_handle(func):
        def magic(*args, **kwargs):
            _self: BasePage = args[0]
            try:
                # return函数执行结果
                return func(*args, **kwargs)
            # 弹窗等异常处理逻辑
            except Exception as e:
                raise e
        return magic
    

    思考:写装饰器的时候,各种return看着有点头晕。每个函数里面都可以return,分别代表什么含义呢???

    def exception_handle(func):
        def magic(*args, **kwargs):
            _self: BasePage = args[0]
            try:
                # 第1处 return:传递func()函数的返回值。如果不写,原有return则失效
                return func(*args, **kwargs)
            # 弹窗等异常处理逻辑
            except Exception as e:
                for element in _self._black_list:
                    elements = _self._driver.find_elements(*element)
                    if len(elements) > 0:
                        elements[0].click()
                        # 异常处理结束,递归继续查找元素 
                        # 第2处 return:递归调用装饰后的函数。magic()表示新函数,func()表示原函数,不可混淆
                        return magic(*args, **kwargs)
                raise e
        # 第3处 return:返回装饰后的函数,装饰器语法。不能返回函数调用magic()
        return magic
    

    装饰器完整实现

    exception.py

    import logging
    
    logging.basicConfig(level=logging.INFO)
    
    
    def exception_handle(func):
        def magic(*args, **kwargs):
            # 防止循环调用报错
            from test_appium.page.base_page import BasePage
            # 获取BasePage实例对象的参数self,这样可以复用driver
            _self: BasePage = args[0]
            try:
                # logging.info('error count is %s' % _self._error_count)
                result = func(*args, **kwargs)
                _self._error_count = 0
                # 返回调用函数的执行结果,要不然返回值会被装饰器吃掉
                return result
            # 弹窗等异常处理逻辑
            except Exception as e:
                # 如果超过最大异常处理次数,则抛出异常
                if _self._error_count > _self._error_max:
                    raise e
                _self._error_count += 1
                for element in _self._black_list:
                    # 用find_elements,就算找不到元素也不会报错
                    elements = _self._driver.find_elements(*element)
                    logging.info(element)
                    # 是否找到弹窗
                    if len(elements) > 0:
                        # 出现弹窗,点击掉
                        elements[0].click()
                        # 弹窗点掉后,重新查找目标元素
                        return magic(*args, **kwargs)
                # 弹窗也没有出现,则抛出异常
                logging.warning("no error is found")
                raise e
        return magic
    

    学习心得

    最好先不看思寒的讲解,根据自己的理解写一遍装饰器,这样学习效果最好。

    遇到问题尝试解决,踩过的坑印象深刻。

    实在没有头绪再参考思寒的解法,那时会有一种豁然开朗的感觉。

    目前就踩到这些坑,如有遗漏,欢迎补充~

    更多技术文章分享及测试资料点此获取

  • 相关阅读:
    datetime和time的时间戳用法
    ER图
    python update()
    理解JWT(JSON Web Token)认证及python实践
    python lambda匿名函数 用法
    flask_restful(转载)
    Flask-SQLALchemy
    创建只有一个元素的元组
    MySQL安装过程中显示无法启动
    聚类
  • 原文地址:https://www.cnblogs.com/hogwarts/p/14109570.html
Copyright © 2020-2023  润新知