自动化测试框架设计-04开发pytest插件

2022-01-18 09:00:00
虫师
转贴:
公众号
2254
摘要:在上一篇文章中我们介绍了4个python单元测试框架,本章我们介绍如何开发pytest插件。

本章我们介绍如何开发pytest插件,在上一篇文章中我们介绍了4个python单元测试框架。大概分两类,一类是必须有类继承的,例如 QTAF 和 unittest, 另一类是可以没有类继承,例如nose/nose2 和 pytest。对于没有可以没有类继承的框架,开发难度会稍大一些。

pytest扩展能力

如果我们需要给pytest增加额外的扩展能力,那么有三种方式。

1. 钩子函数

利用conftest.py 这个特殊的问题,可以创建钩子函数。

  • 目录结构:
pytest_sample/
├── conftest.py
└── test_sample.py 

conftest.py文件中实现如下功能。

import pytest @pytest.fixture def hello():     return "hello 虫师" 

定义一个函数hello(),并使用pytest.fixture装饰器对其进行装饰。fixture的概念我们前面已经做介绍。这里fixture 默认的级别为function,可以理解为被装饰的函数会在每个功能前被执行。

然后,在test_sample.py 测试文件中调动钩子函数。

# 调用钩子函数hello def test_case(hello):     print("hello:", hello)
    assert hello == "hello 虫师" 

在测试用例中钩子函数hello()作为测试用例的参数hello 被调用了。断言 hello函数返回的结果是否为“hello 虫师”

  • 执行用例:
> pytest -vs test_sample.py ================================= test session starts ===========================
collected 1 item

test_sample.py::test_case hello: hello 虫师
PASSED

================================== 1 passed in 2.61s ============================= 

2. 用例装饰器

我们比较常用的用例装饰是parametrize,用法如下。

import pytest @pytest.mark.parametrize(     'a, b', 
    [
        (12),
        (23),
        (34),
    ]
) def test_add(a, b):     print(f'a:{a}, b:{b}')
    assert a + 1 == b 

使用pytest.mark.parametrize()装饰器,装饰test_add()测试用例。定义a和b为参数变量,每次取一组数据进行测试。

  • 运行结果
> pytest -vs test_sample.py ================================= test session starts ===========================
collected 3 items

test_sample.py::test_add[1-2] a:1, b:2
PASSED
test_sample.py::test_add[2-3] a:2, b:3
PASSED
test_sample.py::test_add[3-4] a:3, b:4
PASSED

================================== 3 passed in 2.51s ============================= 

3.命令行参数

通过pytest 命令执行用例的时候作为参数传值。以pytest的扩展插件pytest-base-url 为例。

  • 安装
> pip install pytest-base-url 

编写用例如下:

def test_example(base_url):

    print("base_url:", base_url)

    assert "http" in base_url

这里同样用到了钩子函数base_url, 但base_url的参数定义是在执行用例的时候作为参数传入的。

  • 运行测试
> pytest -vs test_sample.py --base-url https://www.baidu.com ================================= test session starts ===========================
collected 1 item

test_sample.py::test_example base_url: https://www.baidu.com
PASSED

================================== 1 passed in 2.51s ============================= 

pytest扩展插件

实现pytest-hello插件

我么暂且称这个插件为pytest-hello

创建pytest_hello.py文件,实现代码如下:

import pytest



from typing import Any, Optional







def pytest_configure(config: Any) -> None:


    """      register an additional marker

    """

    config.addinivalue_line(











         "markers""env(name): mark test to run only on named environment"



    )

def pytest_runtest_setup(item: Any) -> None:


    """


    Called to perform the setup phase for a test item.

    """

    env_names = [mark.args[0for mark in item.iter_markers(name="env")]

    if env_names:

        if item.config.getoption("--env"not in env_names:

            pytest.skip("test requires env in {!r}".format(env_names))

@pytest.fixture(scope="function")

def hello(hello_name: str) -> str:

    """

    hello Hook function

    """



    return f"hello, {hello_name}"

@pytest.fixture(scope="function") def hello_name(pytestconfig: Any) -> Optional[str]:


    """

    hello_name Hook function




    """     names = pytestconfig.getoption("--hello")

    if len(names) == 0:

        return "虫师"

    if len(names) == 1:

        return names[0]

    return names[0]







def pytest_addoption(parser: Any) -> None:

"""

关键字

发表评论
评论通过审核后显示。
联系我们
  • 联系人:阿道
  • 联系方式:17762006160
  • 地址:青岛市黄岛区长江西路118号青铁广场18楼