Time will tell.
我们在往期的章节中有讲到过fixture
通过scope
参数控制setup
级别。既然有setup
作为用例之前前的操作,用例执行完之后那肯定也有teardown
操作。用到fixture
的teardown
操作并不是独立函数,用yield
关键字呼唤teardown
操作。
一、scope=”module”
1.fixture
参数 scope=”module” ,module
作用是整个.py
文件都会生效,用例调用时,参数写上函数名称就行。
# 新建一个文件test_f1.py
# coding:utf-8
import pytest
@pytest.fixture(scope="module")
def open():
print("打开浏览器,并且打开百度首页")
def test_s1(open):
print("用例1:搜索python-1")
def test_s2(open): # 不传login
print("用例2:搜索python-2")
def test_s3(open):
print("用例3:搜索python-3")
if __name__ == "__main__":
pytest.main(["-s", "test_f1.py"])
运行结果:
============================= test session starts =============================
platform win32 -- Python 3.6.0, pytest-3.6.3, py-1.5.4, pluggy-0.6.0
rootdir: D:\, inifile:
collected 3 items
..\..\..\..\..\..\YOYO\test_f1.py 打开浏览器,并且打开百度首页
.用例1:搜索python-1
.用例2:搜索python-2
.用例3:搜索python-3
========================== 3 passed in 0.01 seconds ===========================
结果,虽然test_s1
,test_s2
,test_s3
三个地方都调用了open
函数,但是它只会在第一个用例前执行一次。
2.如果test_s1
不调用,test_s2
调用open
,test_s3
不调用,运行顺序会怎样?
# 新建一个文件test_f1.py
# coding:utf-8
import pytest
@pytest.fixture(scope="module")
def open():
print("打开浏览器,并且打开百度首页")
def test_s1():
print("用例1:搜索python-1")
def test_s2(open): # 不传login
print("用例2:搜索python-2")
def test_s3():
print("用例3:搜索python-3")
if __name__ == "__main__":
pytest.main(["-s", "test_f1.py"])
运行结果:
============================= test session starts =============================
platform win32 -- Python 3.6.0, pytest-3.6.3, py-1.5.4, pluggy-0.6.0
rootdir: D:\, inifile:
collected 3 items
..\..\..\..\..\..\YOYO\test_f1.py 用例1:搜索python-1
.打开浏览器,并且打开百度首页
.用例2:搜索python-2
.用例3:搜索python-3
========================== 3 passed in 0.01 seconds ===========================
结果,module
级别的fixture
在当前.py
模块里,只会在用例test_s2
第一次调用前执行一次。
二、yield执行teardown
1.前面讲的是在用例前加前置条件,相当于setup
。既然有setup
那就有teardown
,fixture
里面的teardown
用yield
来唤醒teardown
执行。
# 新建一个文件test_f1.py
# coding:utf-8
import pytest
@pytest.fixture(scope="module")
def open():
print("打开浏览器,并且打开百度首页")
yield
print("执行teardown!")
print("最后关闭浏览器")
def test_s1(open):
print("用例1:搜索python-1")
def test_s2(open): # 不传login
print("用例2:搜索python-2")
def test_s3(open):
print("用例3:搜索python-3")
if __name__ == "__main__":
pytest.main(["-s", "test_f1.py"])
运行结果:
============================= test session starts =============================
platform win32 -- Python 3.6.0, pytest-3.6.3, py-1.5.4, pluggy-0.6.0
rootdir: D:\, inifile:
collected 3 items
..\..\..\..\..\..\YOYO\test_f1.py 打开浏览器,并且打开百度首页
.用例1:搜索python-1
.用例2:搜索python-2
.用例3:搜索python-3
执行teardown!
最后关闭浏览器
========================== 3 passed in 0.01 seconds ===========================
2.如果其中一个用例出现异常,不影响yield
后面的teardown
执行,运行结果互不影响,并且在用例全部执行完之后,会呼唤teardown
的内容。
# 新建一个文件test_f1.py
# coding:utf-8
import pytest
@pytest.fixture(scope="module")
def open():
print("打开浏览器,并且打开百度首页")
yield
print("执行teardown!")
print("最后关闭浏览器")
def test_s1(open):
print("用例1:搜索python-1")
# 如果第一个用例异常了,不影响其他的用例执行
raise NameError # 模拟异常
def test_s2(open): # 不传login
print("用例2:搜索python-2")
def test_s3(open):
print("用例3:搜索python-3")
if __name__ == "__main__":
pytest.main(["-s", "test_f1.py"])
运行结果:
\YOYO\test_f1.py 打开浏览器,并且打开百度首页
用例1:搜索python-1
F
open = None
def test_s1(open):
print("用例1:搜索python-1")
# 如果第一个用例异常了,不影响其他的用例执行
> raise NameError # 模拟异常
E NameError
D:\YOYO\test_f1.py:16: NameError
.用例2:搜索python-2
.用例3:搜索python-3
执行teardown!
最后关闭浏览器
3.如果在setup
就异常了,那么是不会去执行yield
后面的teardown
内容。
4.yield
也可以配合with
语句使用,以下是一个官方案例:
# 官方文档案例
# content of test_yield2.py
import smtplib
import pytest
@pytest.fixture(scope="module")
def smtp():
with smtplib.SMTP("smtp.gmail.com") as smtp:
yield smtp # provide the fixture value
三、addfinalizer终结函数
1.除了yield
可以实现teardown
,在request-context
对象中注册addfinalizer
方法也可以实现终结函数。
# 官方案例
# content of conftest.py
import smtplib
import pytest
@pytest.fixture(scope="module")
def smtp_connection(request):
smtp_connection = smtplib.SMTP("smtp.gmail.com", 587, timeout=5)
def fin():
print("teardown smtp_connection")
smtp_connection.close()
request.addfinalizer(fin)
return smtp_connection # provide the fixture value
2.yield
和addfinalizer
方法都是在测试完成后呼叫相应的代码。但addfinalizer
不同的是,它可以注册多个终结函数,这些终结方法总是会被执行,无论在之前的setup code
有没有抛出错误,这个方法对于正确关闭所有的fixture
创建的资源非常便利,即使其一在创建或获取时失败。
好喽,以上就是本章全部内容了,如果你对Python自动化软件测试等更多内容感兴趣可以加入我们扣裙一起学习175317069。有各项测试学习资源发放,更有行业深潜多年的技术人分析讲解。
祝你能成为一名优秀的软件测试工程师!
觉得还不错就【点赞】、【评论】、【关注】吧~
Time will tell.(时间会证明一切)