6. pytest的fixture风格的前后置

前言

  • pytest提供了更加灵活的前后置,通过@pytest.fixture(scope="")来定义不同范围的前后置

1. 如何声明和调用fixture

  • 声明: 使用@pytest.fixture标识的函数即可作为fixture使用
  • 调用: fixture和测试函数都可以调用,只要在函数的入参中直接使用fixture的函数名称即可
    • 代码示例中 outer调用了orderinner,就是fixture调用fixture的例子
    • 代码示例中 test_order调用了orderouter,就是测试函数调用fixture的例子
  • 使用@pytest.mark.usefixtures("fixture_name") 装饰测试函数调用,需要注意的是该方法无法获取fixture的返回值
import pytest


@pytest.fixture
def order():
    return []


@pytest.fixture(autouse=True)
def outer(order, inner):
    order.append("outer")


class TestOne:
    @pytest.fixture
    def inner(self, order):
        order.append("one")

    def test_order(self, order, outer):
        assert order == ["one", "outer"]
  • 熟悉代码的同学可能已经发现上述代码不对,outer定义的范围内inner未定义,理论上调用不同才对,但我们执行代码是正常。因为pytest的执行逻辑是以用例为中心的,顺序应该是先加载用例脚本内容,当加载到测试函数时,执行test_order,发现它需要order函数,于是去调order, 调用结束后调用outer, 而在test_order调用outer时,inner已经被加载了,所以outer可以顺利的调用到inner
  • 看下执行结果
(venv) C:\测试文件夹\project\python\pytest_demo\fixture>pytest -sv test_fixture.py
======================================================= test session starts ========================================================
platform win32 -- Python 3.6.8, pytest-6.2.4, py-1.10.0, pluggy-0.13.1 -- c:\program files (x86)\python36-32\python.exe
cachedir: .pytest_cache
rootdir: C:\测试文件夹\project\python\pytest_demo, configfile: pytest.ini
collected 1 item                                                                                                                    

test_fixture.py::TestOne::test_order PASSED

======================================================== 1 passed in 0.01s =========================================================


2. 调用顺序

  • fixture的调用顺序取决于3个因素
    • 范围:首先执行更高作用域的fixture,从大到小分别为: ["session", "package", "module", "class", "function"]
    • 依赖:当一个夹具请求另一个夹具时,首先执行另一个夹具。例如fixture_a请求fixture_b,fixture_b会先执行,因为a依赖于b它,没有它就不能运行。即使a不需要b的结果,它仍然需要先请求b
    • 自动使用:@pytest.fixture(autouse=True)时为自动使用,同一作用域(scope)内的自动使用的fixture优先使用。需要注意的是,如果fixture_a是自动使用且其依赖于不自动使用的fixture_b时,fixture_b也会变为自动使用,且其作为fixture_a的依赖会比fixture_a提前执行。这也是符合调用依赖的描述的

2.1 范围和依赖

  • 先看范围的执行代码和依赖的执行顺序
# content of test_scope_dep.py
import pytest


@pytest.fixture(scope="session")
def order():
    return []


@pytest.fixture
def func(order):
    order.append("function")


@pytest.fixture(scope="class")
def cls(order):
    order.append("class")


@pytest.fixture(scope="module")
def mod(order):
    order.append("module")


@pytest.fixture(scope="package")
def pack(order):
    order.append("package")


@pytest.fixture(scope="session")
def sess(order):
    order.append("session")


class TestClass:
    def test_order(self, func, cls, mod, pack, sess, order):
        assert order == ["session", "package", "module", "class", "function"]

  • 执行看结果, ordersess作用域都是session,但sess其依赖于order,故order先执行
  • 其余的不同作用域的则按照顺序执行 session > package > module > class > function
(venv) C:\测试文件夹\project\python\pytest_demo\fixture>pytest -sv test_scope_dep.py
======================================================= test session starts ========================================================
platform win32 -- Python 3.6.8, pytest-6.2.4, py-1.10.0, pluggy-0.13.1 -- c:\program files (x86)\python36-32\python.exe
cachedir: .pytest_cache
rootdir: C:\测试文件夹\project\python\pytest_demo, configfile: pytest.ini
collected 1 item                                                                                                                    

test_scope_dep.py::TestClass::test_order PASSED

======================================================== 1 passed in 0.01s =========================================================

2.2. 自动使用

  • 先看代码
@pytest.fixture
def order():
    return []


@pytest.fixture
def a(order):
    order.append("a")


@pytest.fixture
def b(a, order):
    order.append("b")


@pytest.fixture(autouse=True)
def c(order):
    order.append("c")


def test_order(b, order):
    assert order == ["c", "a", "b"]
  • 执行看结果, 所有fixture的作用域都是function,

  • 在测试时仅调用了b, 但因为c是自动使用,故c先执行

  • c依赖于order, 所以是先调用了order再在列表中添加'c'

  • 然后调用b, b依赖于a,故先执行a,并在列表中添加元素'a'

  • 最后执行b,并在列表中添加'b'

  • 执行顺序 order > c > a > b

  • 注意事项

    • 尽管order是其他fixture的依赖项被调用过,但是测试函数中依然要主动调用,不然是无法获取其返回值的。如果order是无返回值的fixture,则无需调用
  • 看到此处,大家可能发现fixture到目前为止也仅仅是描述了前置的信息,那么如何处理teardown

3. fixture声明后置 -- yield关键字

# content of test_fixture_teardown.py
import pytest


@pytest.fixture
def yield_teardown():
    print('\n################  setup part ##############')
    yield True
    print('\n############## teardown part ##############')


def test_yield(yield_teardown):
    print('the case use yield_teardown result is {}'.format(yield_teardown))

  • 在yield关键字前的代码为前置,yield后的代码为后置
  • 执行看下结果,确实按照预期打印
  • 需要注意的是,这种fixture写法中如果不需要返回值时,yield关键字后为空即可,但一定要有该关键字
(venv) C:\测试文件夹\project\python\pytest_demo\fixture>pytest -sv test_fixture_teardown.py
======================================================= test session starts ========================================================
platform win32 -- Python 3.6.8, pytest-6.2.4, py-1.10.0, pluggy-0.13.1 -- c:\program files (x86)\python36-32\python.exe
cachedir: .pytest_cache
rootdir: C:\测试文件夹\project\python\pytest_demo, configfile: pytest.ini
collected 1 item                                                                                                                    

test_fixture_teardown.py::test_yield
################  setup part ##############
the case use yield_teardown result is True
PASSED
############## teardown part ##############


======================================================== 1 passed in 0.01s =========================================================


4. 参数化fixture

  • 假设一种集群测试场景,有两台主机上需要执行同样的用例,那么如果在每个用例上都去写两个参数是不是很麻烦。可不可以在fixture上完成参数化,让同一条用例根据fixture上的参数化执行两次。即通过fixture的参数化实现所有用例的多主机测试。下面以多个测试邮箱连接为例:

  • fixture函数装饰器中增加params参数用于实现参数化

  • fixture函数的形参列表中增加request(固定参数,不可修改),用于接收params传入的参数列表

  • fixture函数的内部调用request.param即可获得列表中的元素

# content of test_fixture_para.py
import pytest
import smtplib


@pytest.fixture(scope="module", params=["smtp.163.com", "smtp.126.com"])
def smtp_connection(request):
    smtp_connection = smtplib.SMTP(request.param, 25, timeout=5)
    yield smtp_connection
    print("\n ############### finalizing {}".format(smtp_connection))
    smtp_connection.close()


def test_fixture_param(smtp_connection):
    print('smtp_connection is {}'.format(smtp_connection))

  • 执行一下用例
(venv) C:\测试文件夹\project\python\pytest_demo\fixture>pytest -sv test_fixture_para.py
======================================================= test session starts ========================================================
platform win32 -- Python 3.6.8, pytest-6.2.4, py-1.10.0, pluggy-0.13.1 -- c:\program files (x86)\python36-32\python.exe
cachedir: .pytest_cache
rootdir: C:\测试文件夹\project\python\pytest_demo, configfile: pytest.ini
collected 2 items                                                                                                                   

test_fixture_para.py::test_fixture_param[smtp.163.com] smtp_connection is <smtplib.SMTP object at 0x03778150>
PASSED
test_fixture_para.py::test_fixture_param[smtp.126.com]
 ############### finalizing <smtplib.SMTP object at 0x03778150>
smtp_connection is <smtplib.SMTP object at 0x037781F0>
PASSED
 ############### finalizing <smtplib.SMTP object at 0x037781F0>


======================================================== 2 passed in 0.69s =========================================================
  • 假想一种接口测试的情况,通常web系统都需要的登录后进行测试,那么一般登录就会放到前置中,但是每个测试者或者每个测试场景中用户名和密码可能都不相同,那么能不能把用户、密码作为参数传给前置函数,并返回认证所需数据呢,用户名和密码这种多个数据就需要使用dict去传参,上代码
@pytest.fixture(params=[{'user': 'user1', 'passwd': '1'}, {'user': 'user2', 'passwd': '2'}])
def login(request):
    user = request.param['user']
    passwd = request.param['passwd']
    token = user + passwd
    return token


def test_login(login):
    print('\nthe token is {}'.format(login))

5. conftest.py: 跨多个文件共享fixture

  • 写到这里有人就会问,这种前置只是在当前文件生效,但登录明显是公共的前置,怎么让其共享呢,name就需要使用conftest.py文件定义fixture
  • 该文件用作为整个目录提供fixture的一种方式。在某个目录的conftest.py文件中定义的fixture可以被该包中的任何测试用例直接使用而无需导入它们(pytest 将自动发现它们)
  • 每个目录都可以有自己的conftest.py,其共享范围为本目录到其所有的递归子目录
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,290评论 6 491
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,107评论 2 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 156,872评论 0 347
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,415评论 1 283
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,453评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,784评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,927评论 3 406
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,691评论 0 266
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,137评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,472评论 2 326
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,622评论 1 340
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,289评论 4 329
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,887评论 3 312
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,741评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,977评论 1 265
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,316评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,490评论 2 348