上期回顾:Poco API精讲之元素属性操作attr、setattr……


以下基于
python3.8;airtestIDE1.2.13;airtest1.2.4;pocoui1.0.85

注意:Poco框架和Airtest框架很多API是同名的,但使用方法完全不一样!!!一定不要搞混了,我初学时也经常搞混,这点一定要注意!
具体Poco框架和Airtest框架是什么关系,可以看之前文章:Airtest Project——UI自动化利器介绍

之前讲了Poco元素定位与属性,今天我们讲下自动化测试中重要的一种操作——等待,自动化脚本健壮性好不好,关键看你等待用的好不好。Poco框架中一共有5种等待方法,分别是wait、wait_for_appearance、wait_for_disappearance、wait_for_any、wait_for_all,下面依次详细看下各个方法怎么用。

1.wait(timeout=3)

等待元素X秒。

参数:
timeout - 等待时间,默认3秒

返回:
自身

源码解析:

# 源码位置:your_python_path\site-packages\poco\proxy.py
    def wait(self, timeout=3):
        start = time.time()
        while not self.exists():
            self.poco.sleep_for_polling_interval()
            if time.time() - start > timeout:
                break
        return self

源码的核心逻辑就是通过while循环,使用Poco框架的exists()方法判断元素是否出现了。
如果出现了则不符合while循环条件退出循环;
如果没出现,等待1.44秒(sleep_for_polling_interval就是等待固定1.44秒)后判断是否达到超时时间,时间达到使用break强制退出循环。
最后不管等没等到,都返回元素对象本身。

我们再来看下exists()的源码,是怎么判断元素是否出现的。

# 源码位置:your_python_path\site-packages\poco\proxy.py
        try:
            return self.attr('visible')
        except (PocoTargetRemovedException, PocoNoSuchNodeException):
            return False

很简单,返回元素的'visible'属性,如果元素出现了,其'visible'属性将是True,即返回True。如果元素没出现,因为属性不存在会报错,所以放在try…except中,并返回False。

一定要注意Poco框架的wait()、exists()和Airtest框架的wait()、exists()是不一样的。可以对比着看看:Airtest API精讲之wait(),exists()

所以Poco框架的wait(),如果等到元素则提前返回;如果在设定时间内没等到也会返回,不会报错。另外wait()之所以返回元素对象本身,是为了代码可以接着写方便。

示例:

from airtest.core.api import *
from poco.drivers.unity3d import UnityPoco

auto_setup(__file__)

poco = UnityPoco()

# 等待元素60秒,如果其出现则立马点击。
# 如果一直没出现,则元素不存在,click会导致PocoNoSuchNodeException报错
poco('测试工程师小站').wait(60).click()


# 等待元素3秒,如果其出现才点击,避免了元素不存在,直接点击导致的报错
qa_button = poco('测试工程师小站')
if qa_button.wait():
    qa_button.click()

2.wait_for_appearance(timeout=120)

等待元素出现,等到提前结束,等不到报错

参数:
timeout - 等待时间,默认120秒

返回:
None

异常:
PocoTargetTimeout:等待元素超时

源码解析:

# 源码位置:your_python_path\site-packages\poco\proxy.py
    def wait_for_appearance(self, timeout=120):
        start = time.time()
        while not self.exists():
            self.poco.sleep_for_polling_interval()
            if time.time() - start > timeout:
                raise PocoTargetTimeout('appearance', self)

通过源码可以看出,wait_for_appearance()和wait()很像,就最后两行代码不一样。区别就是wait_for_appearance()没有返回,且如果没等到元素,会报错。具体使用上的区别看示例。

假设场景:
我们想点击登陆按钮,之后用例才能继续进行,如果登陆不了,用例也就不用执行下去了。在点击登陆按钮之前,有50%几率会弹一个提醒,如果弹了提醒必须点掉,否则就点击不到登陆了。

示例:

from airtest.core.api import *
from poco.drivers.unity3d import UnityPoco

auto_setup(__file__)

poco = UnityPoco()

tips = poco('提醒')
login = poco('测试工程师小站登陆')

# 这里提醒只是几率出现,所以即使不出现,也是正常的,无需报错
if tips.wait(10):  # 等待提醒框最多10秒
    tips.click()  #  如果提醒框出现则点掉,否则影响点击登陆

# login的poco对象定义可以提前定义,定义不会执行
# 只有对象在进行操作时,才会去真正查找,所以Poco适合PO模型
# 这里如果找不到登陆按钮,后面的用例流程也执行不了,所以要报错退出
login.wait_for_appearance()  # 死等登陆120秒,不出现我就报错
login.click()  # login如果出现了,就可以执行该行的点击了

3.wait_for_disappearance(timeout=120)

等待元素消失,消失则提前结束,timeout秒仍未消失则报错

参数:
timeout - 等待时间,默认120秒

返回:
None

异常:
PocoTargetTimeout:等待元素超时

源码解析:

# 源码位置:your_python_path\site-packages\poco\proxy.py
    def wait_for_disappearance(self, timeout=120):
        start = time.time()
        while self.exists():
            self.poco.sleep_for_polling_interval()
            if time.time() - start > timeout:
                raise PocoTargetTimeout('disappearance', self)

wait_for_disappearance的源码就比wait_for_appearance源码少了一个not,就不再多讲了

4.wait_for_any(objects, timeout=120)

等待给定元素列表中任一元素出现,并返回该元素

参数:
objects - 元素列表
timeout - 等待时间

返回:
出现的UIObjectProxy实例

异常:
PocoTargetTimeout:等待元素超时

源码解析:

# 源码位置:your_python_path\site-packages\poco\pocofw.py
    def wait_for_any(self, objects, timeout=120):
        start = time.time()
        while True:
            for obj in objects:
                if obj.exists():
                    return obj
            if time.time() - start > timeout:
                raise PocoTargetTimeout('any to appear', objects)
            self.sleep_for_polling_interval()

等待的源码都是差不多的,wait_for_any就是用了一个for循环去遍历看元素列表中的元素是否有一个出现,只要有1个出现,就立马返回该元素。如果超时了一个没出现,就报错了。

示例:

from airtest.core.api import *
from poco.drivers.unity3d import UnityPoco

auto_setup(__file__)

poco = UnityPoco()

# 屏幕上会随机出现1元、10元、100元,点了就是你的!
# 那妥了,不管出现多少钱,我都点啊,不要白不要,要了还想要!
m1 = poco('monkey1')
m10 = poco('monkey10')
m100 = poco('monkey100')
# 如果1元出现,monkey对象就是1元,如果10元出现,monkey对象就是10元...
monkey = wait_for_any([m1,m10,m100])
monkey.click()

5.wait_for_all(objects, timeout=120)

等待所有元素出现

参数:
objects - 元素列表
timeout - 等待时间

返回:
None

异常:
PocoTargetTimeout:等待元素超时

源码解析:

# 源码位置:your_python_path\site-packages\poco\pocofw.py
    def wait_for_all(self, objects, timeout=120):
        start = time.time()
        while True:
            all_exist = True
            for obj in objects:
                if not obj.exists():
                    all_exist = False
                    break
            if all_exist:
                return
            if time.time() - start > timeout:
                raise PocoTargetTimeout('all to appear', objects)
            self.sleep_for_polling_interval()

核心就是通过for循环去判断元素列表中元素是否都出现了,如果都出现了则返回;如果有一个没出现,就接着找,直到超时报错。
其中while内的前5行代码有一个隐藏的逻辑,就是每次遍历查找所有元素,必须都在了才行。如果第1轮查找,元素1在,元素2没在,第2轮时元素1要重新确认在,防止元素1又消失了。

示例:

from airtest.core.api import *
from poco.drivers.unity3d import UnityPoco

auto_setup(__file__)

poco = UnityPoco()

# 这次规则变了,屏幕上会随机出现1元、10元、100元,但只有1次点击收取按钮的机会
# 为了利益最大化,所以要等所有钱全出现了,再点击收取按钮
m1 = poco('monkey1')
m10 = poco('monkey10')
m100 = poco('monkey100')
get = poco('收取')

wait_for_all([m1,m10,m100])  # 等1元、10元、100元全出现
get.click()

好了,以上就是Poco的5种等待,你学废了吗:)