七、Pytest插件开发

目录

  • 插件的加载方式
  • 什么是hook
  • Pytest有哪些hook函数
  • 如何改写hook函数
  • 实战打包、发布

Pytest插件加载方式

  • 外部插件: pip install安装的插件
  • 本地插件:pytest自动模块发现机制(conftest.py存放的)
  • 内置插件:代码内部的_pytest目录加载
    • 内置插件位置:External Libraries-> site-packages-> _pytest-> hookspec.py
      image.png

什么是hook

  • 把大象装冰箱 ,总共分几步
    1. 打开冰箱门
    2. 把大象装进冰箱
    3. 关闭冰箱门
  • 将以上几步封装成函数


    image.png
  • 如果想在“打开冰箱门”前面加一个“把灯打开”,再“把大象装进冰箱前”“拿出一些零食”,就要重新改变代码
  • hook就是将可能发生的接口/功能/操作预留出来,当想进行这些接口/功能/操作时加入到对应位置即可,即按照流程规范运行。
    image.png

Pytest插件

1、site-package/_pytest/hookspec.py
2、https://docs.pytest.org/en/latest/_modules/_pytest/hookspec.html

实战一:编写自己的插件——编码

  • pytest_collection_modifyitems 收集上来的测试用例实现定制化功能
  • 解决问题:
    • 自定义用例的执行顺序
    • 解决编码问题(中文的测试用例名称)
    • 自动添加标签
  • 含有中文的测试用例名称,改写编码格式:
  • item.name = item.name.encode('utf-8').decode('unicode-escape'):其中name为测试用例的名字
  • item._nodeid = item.nodeid.encode('utf-8').decode('unicode-escape'):其中nodeid为测试用例的路径
  • _pytest / nodes.py
  1. test_chinese.py 代码如下
import pytest

@pytest.mark.parametrize('name', ['哈利','赫敏'])
def test_chinese(name):
    print(name)
  • 运行结果:其中的用例名没有用中文显示,编码格式为Unicode,不支持中文。需要考虑修改测试用例名字和测试用例路径的编码格式
============================= test session starts =============================
rootdir: D:\Programs\DevOps\Python_Practice\Exercises\pytest_plugins
plugins: allure-pytest-2.8.40
collecting ... collected 2 items

test_chinese.py::test_chinese[\u54c8\u5229] 
test_chinese.py::test_chinese[\u8d6b\u654f] 

============================== 2 passed in 0.03s ==============================

Process finished with exit code 0
PASSED                       [ 50%]哈利
PASSED                       [100%]赫敏
  • 修改pytest插件
    1. 在当前目录下新建文件conftest.py
    2. 打开External Libraries-> site-packages-> _pytest-> hookspec.py文件,找到pytest_collection_modifyitems 方法
    def pytest_collection_modifyitems(session: "Session", config: "Config", 
            items: List["Item"] ) -> None:
        """Called after collection has been performed. May filter or re-order
        the items in-place.
    
        :param pytest.Session session: The pytest session object.
        :param _pytest.config.Config config: The pytest config object.
        :param List[pytest.Item] items: List of item objects.
        """
    
    1. 将方法复制到 conftest.py 文件下进行改写
    def pytest_collection_modifyitems(ession, config, items):
     for item in items:
         item.name = item.name.encode('utf-8').decode('unicode-escape')
         item._nodeid = item.nodeid.encode('utf-8').decode('unicode-escape')
    
    1. 再次执行test_chinese.py 结果如下:发现用例名改为了中文
rootdir: D:\Programs\DevOps\Python_Practice\Exercises\pytest_plugins
plugins: allure-pytest-2.8.40
collecting ... collected 2 items

test_chinese.py::test_chinese[哈利] PASSED                               [ 50%]哈利

test_chinese.py::test_chinese[赫敏] PASSED                               [100%]赫敏


============================== 2 passed in 0.02s ==============================

Process finished with exit code 0

实战二:倒序执行用例

  • 由于def pytest_collection_modifyitems(session, config, items:List): 中items为用例列表,所以可以对其进行列表的方法,比如.reverse()进行倒序执行
  • conftest.py文件中加入items.reverse()
from typing import List

def pytest_collection_modifyitems(session, config, items:List):
    # 修改编码
    for item in items:
        item.name = item.name.encode('utf-8').decode('unicode-escape')
        item._nodeid = item.nodeid.encode('utf-8').decode('unicode-escape')

    # 修改用例执行顺序,其中 items 就是所有用例列表
    items.reverse()  # 倒序执行
  • 再次执行test_chinese.py 结果如下:发现用例执行顺序改变
rootdir: D:\Programs\DevOps\Python_Practice\Exercises\pytest_plugins
plugins: allure-pytest-2.8.40
collecting ... collected 2 items

test_chinese.py::test_chinese[赫敏] PASSED                               [ 50%]赫敏

test_chinese.py::test_chinese[哈利] PASSED                               [100%]哈利


============================== 2 passed in 0.02s ==============================

Process finished with exit code 0

实战三:只执行打了标签的测试用例

  • 测试代码如下,其中有两条用例名包含login
import pytest

@pytest.mark.parametrize('name', ['哈利','赫敏'])
def test_chinese(name):
    print(name)

def test_login():
    print("login")

def test_login_fail():
    print("login fail")
    assert False

def test_search():
    print("search")
  • 修改conftest.py 文件,给login打上标签
from typing import List
import pytest

def pytest_collection_modifyitems(session, config, items:List):
    # 修改编码
    for item in items:
        item.name = item.name.encode('utf-8').decode('unicode-escape')
        item._nodeid = item.nodeid.encode('utf-8').decode('unicode-escape')

        # 如果login在测试用例路径中,则对其打标签
        if "login" in item.nodeid:
            item.add_marker(pytest.mark.login)

    # 修改用例执行顺序,其中 items 就是所有用例列表
    items.reverse()  # 倒序执行
  • 在Terminal中执行pytest -m login -vs 其中-m 为运行指定标签的用例,后面跟上标签名,-v为打印详细信息,-s用于显示测试函数中print()函数输出。执行结果如下,可以看到只执行了login标签的用例
(venv) D:\Programs\DevOps\Python_Practice\Exercises\pytest_plugins>pytest -m login -vs
============================================================================================= test session starts =============================================================================================
platform win32 -- Python 3.8.5, pytest-6.2.2, py-1.10.0, pluggy-0.13.1 -- d:\programs\devops\python_practice\venv\scripts\python.exe
cachedir: .pytest_cache
rootdir: D:\Programs\DevOps\Python_Practice\Exercises\pytest_plugins
plugins: allure-pytest-2.8.40
collected 5 items / 3 deselected / 2 selected                                                                                                                                                                  

test_chinese.py::test_login_fail login fail
FAILED
test_chinese.py::test_login login
PASSED

================================================================================================== FAILURES ===================================================================================================
_______________________________________________________________________________________________ test_login_fail _______________________________________________________________________________________________

    def test_login_fail():
        print("login fail")
>       assert False
E       assert False

test_chinese.py:17: AssertionError
============================================================================================== warnings summary ===============================================================================================
conftest.py:17
  D:\Programs\DevOps\Python_Practice\Exercises\pytest_plugins\conftest.py:17: PytestUnknownMarkWarning: Unknown pytest.mark.login - is this a typo?  You can register custom marks to avoid this warning - for d
etails, see https://docs.pytest.org/en/stable/mark.html
    item.add_marker(pytest.mark.login)

-- Docs: https://docs.pytest.org/en/stable/warnings.html
=========================================================================================== short test summary info ===========================================================================================
FAILED test_chinese.py::test_login_fail - assert False
============================================================================ 1 failed, 1 passed, 3 deselected, 1 warning in 0.23s =============================================================================

实战四:添加命令行参数

def pytest_addoption(parser):
  mygroup = parser.getgroup("hogwarts") #group将下面所有的option都展示在这个group下。
  mygroup.addoption("--env",            #注册一个命令行选项
                    default='test',     #参数的默认值
                    dest='env'          ,#存储的变量
                    help='set your run env' #帮助提示参数的描述信息
                    )
  • 如何针对传入的不同参数完成不同的逻辑处理?创建一fixture
@pytest.fixture(scope='session')
def cmdoption(request):
  return request.config.getoption("--env", default='test')
  • 在terminal中执行pytest --help就能发现自定义的命令行参数:
    image.png
  • 如果要获取addoption中定义的命令行参数,可以在conftest.py中定义fixture如下:
# 定义fixture从而获取addoption里面函数
@pytest.fixture(scope='session')
def cmdoption(request):
    env = request.config.getoption("--env", default='test')
    if env == 'test':
        print("这是测试环境")
    elif env == 'dev':
        print("这是开发环境")
  • 新建一个测试用例如下,将在conftest中定义的fixture函数名传过来
# 将在conftest中定义的fixture函数传过来
def test_env(cmdoption):
    print(cmdoption)
  • 执行结果如下:
test_chinese.py::test_env 这是测试环境
PASSED                                         [100%]test


============================== 1 passed in 0.04s ==============================

Process finished with exit code 0
  • 如果要修改环境为开发环境,则在Terminal中执行pytest --env dev test_chinese.py::test_env -vs,执行结果如下:
rootdir: D:\Programs\DevOps\Python_Practice\Exercises\pytest_plugins
plugins: allure-pytest-2.8.40
collected 1 item                                                                                                                                                                                               

test_chinese.py::test_env 这是开发环境
dev
PASSED

实战五:获取环境数据

  • 添加环境数据


    image.png
    • 其中dev下的datas.yaml内容如下:
env:
 host: https://www.baidu.com
 port: 443
  • 其中test下的datas.yaml内容如下:
env:
 host: http://www.baidu.com
 port: 80
  • 修改conftest.py文件如下:主要修改cmdoption函数
from typing import List
import pytest
import yaml


def pytest_collection_modifyitems(session, config, items:List):
    # 修改编码
    for item in items:
        item.name = item.name.encode('utf-8').decode('unicode-escape')
        item._nodeid = item.nodeid.encode('utf-8').decode('unicode-escape')

        # 如果login在测试用例路径中,则对其打标签
        if "login" in item.nodeid:
            item.add_marker(pytest.mark.login)

    # 修改用例执行顺序,其中 items 就是所有用例列表
    items.reverse()  # 倒序执行

# 添加一个命令行参数
def pytest_addoption(parser):
    mygroup = parser.getgroup("hogwarts") #group将下面所有的option都展示在这个group下。
    mygroup.addoption("--env",            #注册一个命令行选项
                    default='test',     #参数的默认值
                    dest='env'          ,#存储的变量
                    help='set your run env' #帮助提示参数的描述信息
                    )

# 定义fixture从而获取addoption里面函数
@pytest.fixture(scope='session')
def cmdoption(request):
    env = request.config.getoption("--env", default='test')

    if env == 'test':
        print("这是测试环境")
        datapath = "./datas/test/datas.yml"

    elif env == 'dev':
        print("这是开发环境")
        datapath = "./datas/dev/datas.yml"

    with open(datapath) as f:
        datas = yaml.safe_load(f)
    return env, datas
  • 修改测试用例如下:
# 将在conftest中定义的fixture函数传过来
def test_env(cmdoption):
    env, datas = cmdoption
    print(datas)
    host = datas['env']['host']
    port = datas['env']['port']
    url = str(host) + ":" + str(port)
    print(url)
  • 测试结果:
plugins: allure-pytest-2.8.40
collected 1 item

test_chinese.py::test_env 这是开发环境
{'env': {'host': 'https://www.baidu.com', 'port': 443}}
https://www.baidu.com:443
PASSED

打包发布

  • 打包必须要有代码和setup.py 文件
  • setup.py 是一个构建工具


    image.png

打包需要两个工具

  • wheel, setuptools
  • setup.py 文件
  • 目录结构


    image.png
  1. setup.py 文件:
from setuptools import setup
setup(
    name='pytest_encode',
    url='https://github.com/xxx/pytest-encode',
    version='1.0',
    author="loafer",
    author_email='418974188@qq.com',
    description='set your encoding and logger',
    long_description='Show Chinese for your mark.parametrize(). Define logger variable for getting your log',
    classifiers=[# 分类索引 ,pip 对所属包的分类
        'Framework :: Pytest',
        'Programming Language :: Python',
        'Topic :: Software Development :: Testing',
        'Programming Language :: Python :: 3.8',
    ],
    license='proprietary',
    packages=['pytest_encode'],
    keywords=[
        'pytest', 'py.test', 'pytest_encode',
    ],

    # 需要安装的依赖
    install_requires=[
        'pytest'
    ],
    # 入口模块 或者入口函数
    entry_points={
        'pytest11': [
            'pytest-encode = pytest_encode',
        ]
    },
    zip_safe=False
)
  1. pytest_encode 中的 _init_.py 文件
from typing import List

def pytest_collection_modifyitems(session, config, items:List):
    # 修改编码
    for item in items:
        item.name = item.name.encode('utf-8').decode('unicode-escape')
        item._nodeid = item.nodeid.encode('utf-8').decode('unicode-escape')

    # 修改用例执行顺序,其中 items 就是所有用例列表
    items.reverse()  # 倒序执行
  1. test_encode.py 文件
import pytest

@pytest.mark.parametrize('name', ['哈利','赫敏'])
def test_chinese(name):
    print(name)
  1. 安装wheel 工具:pip install wheel
  • 打包命令:
    python setup.py sdist bdist_wheel
  • 打包完后


    image.png
  • dist 中上面的是源码包,下面的是whl包,可以通过pip install 进行安装

发布

image.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,384评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,845评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,148评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,640评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,731评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,712评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,703评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,473评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,915评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,227评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,384评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,063评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,706评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,302评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,531评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,321评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,248评论 2 352

推荐阅读更多精彩内容