Python单元测试之pytest

x33g5p2x  于2021-10-09 转载在 Python  
字(6.0k)|赞(0)|评价(0)|浏览(371)

前提:需要安装pytest和pytest-html(生成html测试报告)
pip install pytest
pip install pytest-html

安装成功展示:

1.命名规则

Pytest单元测试中的类名和方法名必须是以test开头,执行中只能找到test开头的类
和方法,比unittest 更加严谨

setup和teardown运行于测试方法的始末,即:运行一次测试函数会运行一次

setup_class和 teardown_class运行于测试方法的始末,但是不管有多少测试函数都只执行一次
案例:

import pytest

class TestClass:
    def setup_class(self):
        print('测试前执行一次')
    def setup(self):
        print('start')
    def test01(self):
        print('test01')
    def test02(self):
        print('test02')
    def teardown(self):
        print('end')
    def teardown_class(self):
        print('测试后执行一次')
if __name__ == '__main__':
    pytest.main(['-s'])

2.Pytest生成自带的html测试报告

前提条件:需要下载pytest-html模块(python自带的生成测试报告模块)

pip install pytest-html

案例一:

pytest.main(“模块.py”)【运行指定模块下,运行所有test开头的类和测试用例】

pytest.main(["--html=./report.html","模块.py"])

案例二:

运行指定模块指定类指定用例,冒号分割,并生成测试报告

pytest.main([‘--html=./report.html’,‘模块.py::类名::用例名'])

案例三:

直接执行pytest.main() 【自动查找当前目录下,以test开头的文件或者以test结尾的py文件】

pytest.main([‘--html=./report.html’])

如果没有找到以test开头的文件或者以test结尾的py文件展示如下图:

pytest调用语句

pytst.main(['-x','--html=./report.html'])

-x出现一条测试用例失败就退出测试

-x出现一条测试用例失败就退出测试 
-v: 丰富信息模式, 输出更详细的用例执行信息 
-s:显示print内容 
-q: 简化结果信息,不会显示每个用例的文件名

扩展:跳充

使用@pytest.mark.skip()跳过该用例(函数)

跳过test01用例展示:

3.pytest的运行方式

. 点号,表示用例通过 
F 表示失败 Failure 
E 表示用例中存在异常 Error
s 表示跳过用例

4.文件读取

读取csv文件

import csv #导入csv模块 
class ReadCsv(): 
	def read_csv(self): 
	item =[] #定义一个空列表 
	c = csv.reader(open("../commonDemo/test1.csv","r")) #得到csv文件对象 
	for csv_i in c: 
		item.append(csv_i) #将获取的数据添加到列表中 
		return item 
r = ReadCsv() 
print(r.read_csv())

读取xml文件

from xml.dom import minidom 
class Readxml(): 
	def read_xml(self,filename,onename,twoname): 
		root =minidom.parse(filename) 
		firstnode =root.getElementsByTagName(onename)[0]
		secondnode=firstnode.getElementsByTagName(twoname)[0].firstChild.data 
		return secondnode

5.allure

allure是一款轻量级并且非常灵活的开源测试报告框架。 它支持绝大多数测试框架, 例如TestNG、Pytest、JUint等。它简单易用,易于集成。

首先配置allure的环境变量

验证allure是否配置成功

其次要安装allure

pip install allure-pytest

allure-pytest是Pytest的一个插件,通过它我们可以生成Allure所需要的用于生成测试报告的数据

allure常用的几个特性

@allure.feature # 用于描述被测试产品需求 
@allure.story # 用于描述feature的用户场景,即测试需求 
with allure.step: # 用于描述测试步骤,将会输出到报告中 
allure.attach # 用于向测试报告中输入一些附加的信息,通常是一些测试数据,
截图等

案例一:实现用户登录功能,场景为登录成功和登录失败

import pytest,allure,os  # 导入模块

class TestAnli:
    @allure.feature('用户登录功能') # 用于定义被测试的功能,被测产品的需求点
    @allure.story('用户登录成功') # 用于定义被测功能的用户场景,即子功能点
    def test_success(self):
        assert 1==1
    @allure.feature('用户登录功能') # 用于定义被测试的功能,被测产品的需求点
    @allure.story('用户登录失败') # 用于定义被测功能的用户场景,即子功能点
    def test_fail(self):
        assert 1==2

if __name__ == '__main__':
    pytest.main(['--alluredir', 'report/result','test_denglu.py'])  # 生成json类型的测 试报告
    split = 'allure ' + 'generate ' + './report/result ' + '-o ' + './report/html ' + '--clean' #将测试报告转为html格式
    os.system(split) # system函数可以将字符串转化成命令在服务器上运行

pytest和allure效果展示

案例二:实现产品信息展示,车展中的各种车的品牌

import pytest,allure,os  # 导入模块

class TestCart:
    @allure.feature('车展') # 用于描述被测试产品需求
    @allure.story('展示车辆') # 用于描述feature的用户场景,即测试需求
    def test_cart(self):
        with allure.step('查看大众系列的车'): # 用于描述测试步骤,将会输出到报告中
            allure.attach('迈腾') # 用于向测试报告中输入一些附加的信息
            allure.attach('辉腾')  # 用于向测试报告中输入一些附加的信息
        with allure.step('查看奔驰系列的车'): # 用于描述测试步骤,将会输出到报告中
            allure.attach('大G') # 用于向测试报告中输入一些附加的信息
            allure.attach('迈巴赫')  # 用于向测试报告中输入一些附加的信息

if __name__ == '__main__':
    pytest.main(['--alluredir', 'report/result', 'test_cart.py'])  # 生成json类型的测 试报告
    split = 'allure ' + 'generate ' + './report/result ' + '-o ' + './report/html ' + '--clean'  # 将测试报告转为html格式
    os.system(split)  # system函数可以将字符串转化成命令在服务器上运行

pytest和allure效果展示

请欣赏福利

案例:
通过读取csv/xml数据并且结合使用allure展示测试报告,验证开发中的add()
和reduct()操作(在@allure.story分别实现相加减)

第一种:xml文件读取数据进行单元测试并生成测试报告

第一步:开发的代码

class Demo:
    def add(self,a,b):
        c=a+b
        return c
    def reduct(self,a,b):
        c=a-b
        return c

第二步:创建一个xml文件

<node>
    <add>
        <add1>2</add1>
        <add2>1</add2>
        <add3>3</add3>
    </add>
    <reduct>
        <reduct1>5</reduct1>
        <reduct2>2</reduct2>
        <reduct3>3</reduct3>
    </reduct>
</node>

第三步:读取xml文件

from xml.dom import minidom
class Readxml():
    def read_xml(self,filename,onename,twoname):
        root =minidom.parse(filename)
        firstnode =root.getElementsByTagName(onename)[0]
        secondnode=firstnode.getElementsByTagName(twoname)[0].firstChild.data
        return secondnode
r = Readxml()

第四步:传递参数进行单元测试并结合allure生成测试报告

import pytest,allure,os
from demo.demo import Demo
from readdata.readxml import Readxml

r=Readxml()
add1=r.read_xml('../data/testxml.xml','add','add1')
add2=r.read_xml('../data/testxml.xml','add','add2')
add3=r.read_xml('../data/testxml.xml','add','add3')
re1=r.read_xml('../data/testxml.xml','reduct','reduct1')
re2=r.read_xml('../data/testxml.xml','reduct','reduct2')
re3=r.read_xml('../data/testxml.xml','reduct','reduct3')
d=Demo()

class TestFuli:
    @allure.feature('相加的操作')
    @allure.story('相加成功')
    def test_add(self):
        res=d.add(int(add1),int(add2))
        assert res==int(add3)

    @allure.feature('相减的操作')
    @allure.story('相减成功')
    def test_reduct(self):
        res = d.reduct(int(re1), int(re2))
        assert res == int(re3)
if __name__ == '__main__':
    pytest.main(['--alluredir', 'report/result', 'test_lianxi.py'])  # 生成json类型的测试报告
    split = 'allure ' + 'generate ' + './report/result ' + '-o ' + './report/html ' + '--clean'  # 将测试报告转为html格式
    os.system(split)  # system函数可以将字符串转化成命令在服务器上运行

展示结果:

第二种:csv文件读取数据进行单元测试并生成测试报告

第一步:开发的代码

class Demo:
    def add(self,a,b):
        c=a+b
        return c
    def reduct(self,a,b):
        c=a-b
        return c

第二步:创建两个csv文件

2,4,6
6,1,5

第三步:读取csv文件

import csv  # 导入csv模块

class ReadCsv():
    def add_csv(self):
        item = []  # 定义一个空列表
        c = csv.reader(open("../data/testcsv1.csv", "r"))  # 得到csv文件对象
        for csv_i in c:
            item.append(csv_i)  # 将获取的数据添加到列表中
        return item
    
    def reduct_csv(self):
        item = []  # 定义一个空列表
        c = csv.reader(open("../data/testcsv2.csv", "r"))  # 得到csv文件对象
        for csv_i in c:
            item.append(csv_i)  # 将获取的数据添加到列表中
        return item

r = ReadCsv()

第四步:传递参数进行单元测试并结合allure生成测试报告

import pytest,allure,os
from demo.demo import Demo
from readdata.readcsv import ReadCsv

r=ReadCsv()
add1=r.add_csv()
re1=r.reduct_csv()
d=Demo()
class TestFuli:
    @allure.feature('相加的操作')
    @allure.story('相加成功')
    def test_add(self):
        for i in add1:
            res=d.add(int(i[0]),int(i[1]))
            assert res==int(i[2])

    @allure.feature('相减的操作')
    @allure.story('相减成功')
    def test_reduct(self):
        for i in re1:
            res=d.reduct(int(i[0]),int(i[1]))
            assert res==int(i[2])
if __name__ == '__main__':
    pytest.main(['--alluredir', 'report/result', 'test_xml.py'])  # 生成json类型的测试报告
    split = 'allure ' + 'generate ' + './report/result ' + '-o ' + './report/html ' + '--clean'  # 将测试报告转为html格式
    os.system(split)  # system函数可以将字符串转化成命令在服务器上运行

相关文章