Pytest框架 — 08、Pytest的Fixture(部分前后置)(三)

锦通  金牌会员 | 2022-9-16 17:19:40 | 显示全部楼层 | 阅读模式
打印 上一主题 下一主题

主题 814|帖子 814|积分 2442

目录

8、Fixture带返回值

在fixture中我们可以使用yield或者return来返回我们需要的东西,如测试数据,数据库连接对象,文件对象等。

  • 没有后置处理
    直接采用return的方式返回数据(yield也可以)
  1. import pytest
  2. @pytest.fixture()
  3. def data_fixture():
  4.     return ["a", "b", "c"]
  5. def test_case(data_fixture):
  6.     print(f"测试数据为:{data_fixture}")
  7. """
  8. 执行结果
  9. fixture/yield_data.py::test_case 测试数据为:['a', 'b', 'c']
  10. PASSED
  11. """
复制代码

  • 有后置处理


  • 使用yield返回数据,在yield下面添加后置处理
  1. import pytest
  2. @pytest.fixture()
  3. def data_fixture():
  4.    print("【前置】")
  5.    yield ["a", "b", "c"]
  6.    print("【后置】")
  7. def test_case(data_fixture):
  8.    print(f"测试数据为:{data_fixture}")
  9. """
  10. 执行结果
  11. fixture/yield_data.py::test_case 【前置】
  12. 测试数据为:['a', 'b', 'c']
  13. PASSED【后置】
  14. """
复制代码

  • 使用return返回数据,通过addfinalizer注册后置处理
  1. import pytest
  2. @pytest.fixture()
  3. def data_fixture(request):
  4.     print("【前置】")
  5.     def addfinalizer_demo():
  6.         print("【后置】")
  7.     request.addfinalizer(addfinalizer_demo)
  8.     return ["a", "b", "c"]
  9. def test_case(data_fixture):
  10.     print(f"测试数据为:{data_fixture}")
  11. """
  12. 执行结果
  13. fixture/return_data.py::test_case 【前置】
  14. 测试数据为:['a', 'b', 'c']
  15. PASSED【后置】
  16. """
复制代码
9、Fixture实现参数化

可以通过fixture的params参数来实现参数化。(有更好的参数化方式,故此方法不常用)
示例1:
  1. import pytest
  2. data = [("张三","男"), ("李四","女")]
  3. @pytest.fixture(params=data)
  4. def fixture_data(request):
  5.     print("【前置】")
  6.     def addfinalizer_demo():
  7.         print("【后置】")
  8.     request.addfinalizer(addfinalizer_demo)
  9.     return request.param
  10. def test_case(fixture_data):
  11.     print(fixture_data[0], fixture_data[1])
  12. """
  13. 执行结果
  14. fixture/fixture_params.py::test_case[fixture_data0] 【前置】
  15. 张三 男
  16. PASSED【后置】
  17. fixture/fixture_params.py::test_case[fixture_data1] 【前置】
  18. 李四 女
  19. PASSED【后置】
  20. """
复制代码
示例2:通过ids参数为每个用例取别名
  1. import pytest
  2. data = [("张三","男"), ("李四","女")]
  3. @pytest.fixture(params=data,ids=["Developer", "Tester"])
  4. def fixture_data(request):
  5.     print("【前置】")
  6.     def addfinalizer_demo():
  7.         print("【后置】")
  8.     request.addfinalizer(addfinalizer_demo)
  9.     return request.param
  10. def test_case(fixture_data):
  11.     print(fixture_data[0], fixture_data[1])
  12. """
  13. 执行结果
  14. fixture/fixture_params_ids.py::test_case[Developer] 【前置】
  15. 张三 男
  16. PASSED【后置】
  17. fixture/fixture_params_ids.py::test_case[Tester] 【前置】
  18. 李四 女
  19. PASSED【后置】
  20. """
复制代码
总结:

  • 通过params参数实现参数化需要用到request关键字,通过return request.param将测试数据返回。
  • params中有多少数据,ids就必须对应多少参数,脱离使用params实现参数化,ids将毫无意义。
  • params支持的参数格式:

    • 列表[]
    • 元祖()
    • 元祖列表[(),(),()]
    • 字典列表[{},{},{}]
    • 字典元祖({},{},{})

10、可靠的Fixture写法

假如,我们有一个登录页面,需要进行登录测试。为了方便测试,我们还有一个管理员的api,可以直接调用来生成测试用户。那么,这个测试场景通常会这样去构建:

  • 通过管理API创建一个用户
  • 使用Selenium启动浏览器
  • 进入我们网站的登录页面
  • 使用创建好的用户进行登录
  • 断言登录后的用户名出现在登录页的页眉中
(一)一个不可靠Fixture的例子

说明:该例子不可执行,仅做示例
  1. import pytest
  2. from selenium import webdriver
  3. import AdminApiClent
  4. import LoginPage
  5. import IndexPage
  6. from urllib.parse import urljoin
  7. base_url = "https:xxx.com"
  8. username = "xxx"
  9. @pytest.fixture()
  10. def setup_fixture():
  11.     # 【前置处理】
  12.     # 1、通过AdminApiClent创建一个用户
  13.     client = AdminApiClent()
  14.     client.create_user(username)
  15.     # 2、获取驱动对象
  16.     driver = webdriver.Chrome()
  17.     # 3、打开登录页
  18.     driver.(urljoin(base_url,"/login"))
  19.     # 4、执行登录操作
  20.     login_page = LoginPage(driver)
  21.     login_page.login(username)
  22.     yield username
  23.     # 【后置处理】
  24.     # 1、关闭浏览器
  25.     driver.quit()
  26.     # 2、删除用户
  27.     client.del_user(username)
  28.     # 3、退出AdminApiClent
  29.     client.quit()
  30. def test_login_success(setup_fixture):
  31.     # 断言登录后用户名是否出现在首页
  32.     assert setup_fixture == IndexPage.username
复制代码
上面例子存在的问题:

  • 在setup_fixture这一个fixture函数中做的事情太多,很多步骤不容易重用。
  • 假设setup_fixture中yield之前出现异常,后置处理不会执行。(虽然可以使用addfinalizer,但是不易维护和重用)
(二)修改成可靠Fixture的例子

说明:该例子不可执行,仅做展示
  1. import pytest
  2. from selenium import webdriver
  3. import AdminApiClent
  4. import LoginPage
  5. import IndexPage
  6. from urllib.parse import urljoin
  7. base_url = "https:xxx.com"
  8. username = "xxx"
  9. # 处理AdminApiClient
  10. @pytest.fixture(name="client")
  11. def AdminApiClient_fixture():
  12.     client = AdminApiClent()
  13.     yield client
  14.     client.quit()
  15. # 处理用户
  16. @pytest.fixture()
  17. def user_fixture(client):
  18.     client.create_user(username)
  19.     yield username
  20.     client.del_user(username)
  21. # 处理驱动
  22. @pytest.fixture()
  23. def driver_fixture():
  24.     driver = selenium.Chrome()
  25.     yield driver
  26.     driver.quit()
  27. # 处理登录
  28. @pytest.fixture()
  29. def login_fixture(driver_fixture, user_fixture):
  30.     driver_fixture.get(urljoin(base_url, "/login"))
  31.     login_page = LoginPage(driver_fixture)
  32.     login_page.login(user_fixture)
  33. # 处理首页
  34. @pytest.fixture()
  35. def index_fixture(driver_fixture):
  36.     return IndexPage(driver_fixture)
  37. # 断言
  38. def test_login_success(login_fixture,index_fixture,user_fixture):
  39.     assert user_fixture == index_fixture.username
复制代码
这么改造的优点:

  • 每一步都单独进行了封装,提高了复用性
  • 假设运行中user_fixture报错了,那么不影响driver_fixture关闭浏览器
  • 假设运行中driver_fixture报错了,那么根本就不会执行user_fixture
参考
https://www.cnblogs.com/liuyuelinfighting/p/15999810.html
https://zhuanlan.zhihu.com/p/359125816

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!
回复

使用道具 举报

0 个回复

正序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

锦通

金牌会员
这个人很懒什么都没写!

标签云

快速回复 返回顶部 返回列表