百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

pytest测试框架+allure插件自定义测试报告

zhezhongyun 2024-12-15 17:54 64 浏览

之前文章我们学习了pytest-html插件生成html测试报告,本章介绍另一个好用的插件。allure-pytest 是pytest的一个插件,帮助我们再pytest测试框架中集成Allure框架来生成详细的测试报告。

Allure 是一个灵活的轻量级多语言测试报告工具,支持包括 pytest /testng在内的多种测试框架,能够生成易于理解和丰富的测试报告。

allure-pytest能够生成包含测试步骤、日志、标题、优先级、附件等详细信息的测试报告,有助于更好地理解测试用例的执行情况。

自定义功能:支持通过装饰器(比如@allure.epic、@allure.feature、@allure.story、@allure.title、@allure.description 等)来添加额外的测试信息,如功能模块、用例标题、严重程度等,从而生成更加详细和有用的报告。

官方文档:https://allurereport.org/docs/pytest/

allure-pytest插件安装

使用pip命令安装: pip install allure-pytest

(安装在pytest工程所运行的python环境,python的安装目录或者虚拟环境目录,可以参考之前文章查看运行环境pycharm配置pytest运行环境)

安装和配置allure命令行工具(mac系统)

以下以mac电脑举例:

1)下载和配置jdk

下载地址:Java Downloads | Oracle,选择适用于Mac OS的JDK版本,点击下载即可。
下载完之后,双击安装包按照如下页面指导直接安装即可

默认安装目录:/Library/Java/JavaVirtualMachines,打开终端,进入目录可以看到jdk程序。

配置环境变量:使用open ~/.bash_profile打开变量文件,不存在先创建一个该文件。

vi命令打开后,添加如下信息保存退出

export JAVA_HOME=/Library/Java/JavaVirtualMachines/jdk-22.jdk/Contents/Home

执行source ~/.bash_profile使配置生效,然后执行java --version可以看到java版本了。

2)下载命令行工具

进入以下地址下载allure工具

https://repo.maven.apache.org/maven2/io/qameta/allure/allure-commandline/

最新的版本是2.9.0

3)配置allure命令行工具

将下载的allure包解压,打开终端,配置环境变量执行vi ~/.bash_profile然后添加export PATH="/Users/aaa/Downloads/allure-2.9.0/bin:$PATH",填写的是allure的目录信息。

执行source ~/.bash_profile使配置生效,然后在任意目录执行allure --version可以看到allure执行成功了。

生成测试报告

测试用例执行后,生成最简易的测试报告,步骤如下:

1)使用时无需导入,直接在用例执行命令中添加参数

执行带上参数--alluredir=./allure-results,参数后面跟着用例结果的目录。

执行后在目标目录中生成的是一堆json文件。

2)将测试结果生成html测试报告

  • 可以生成在线测试报告并自动通过浏览器打开

命令:allure serve allure-results

执行后自动打开浏览器,展示测试报告内容。生成的报告位置见日志打印

  • 自定义生成测试报告的目录,手动在浏览器打开

命令:allure generate allure-results -o allure-report --clean

-o后面跟着报告的目录

--clean表示先清楚该目录内容再生成测试报告,否则会提示目录已被使用。


测试报告如下,比插件pytest-html生成的报告内容更丰富,页面更好看。

Allure报告提供了丰富的信息和功能,展示测试用例执行结果,用例的层级结构,附件和截图,测试指标和趋势图,还有其他的标记、过滤、定制等,让测试结果更加直观和易于理解。

allure的装饰器功能

使用前先导入 import allure

官方文档:https://allurereport.org/docs/pytest-reference/#metadata

装饰器和方法

参数说明

@allure.epic(*epics: str)

@allure.feature(*features: str)

@allure.story(*stories: str)


allure.dynamic.epic(*epics: str)

allure.dynamic.feature(*features: str)

allure.dynamic.story(*stories: str)

定义用例的层级结构,对应敏捷中的需求/特性/故事,每个用例都可以定义多个装饰器或者方法。


@allure.parent_suite(parent_suite_name)

@allure.suite(suite_name)

@allure.sub_suite(sub_suite_name)


allure.dynamic.parent_suite(parent_suite_name)

allure.dynamic.suite(suite_name)

allure.dynamic.sub_suite(sub_suite_name)


定义用例的套件结构:为测试分配父套件、套件或子套件的名称


@allure.title(test_title: str)


allure.dynamic.title(test_title: str)

定义测试用例或者fixture方法的标题

如果测试函数有参数,也支持将参数传入标题中。

@allure.step(title: str)

with allure.step(title: str)--上下文管理器

定义测试用例的步骤

@allure.description(test_description: str)

allure.dynamic.description(test_description: str)

定义测试用例的描述信息。如果没有提供,Allure将使用测试函数的描述。

@allure.tag(*tags: str)

allure.dynamic.tag(*tags: str)

设置测试用例的标记,一次可以接受多个标记。

@allure.severity(severity_level: str)


allure.dynamic.severity(severity_level: str)

设置测试用例的严重性,枚举值:“trivial”···, “minor”, “normal”, “critical”, and “blocker”.

@allure.label(label_type: str, *labels: str)


allure.dynamic.label(label_type: str, *labels: str)

设置测试用例的标签。

该函数的第一个参数是标签名称。它可以是来自LabelType类或任何其他字符串的常量。

第二个和所有剩余的参数都是要添加的值。对于每个值,该函数将添加一个带有给定名称和给定值的标签。

@allure.link(url:str,link_type:str=LinkType.LINK, name: str = None)

@allure.issue(url: str, name: str = None)

@allure.testcase(url: str, name: str = None)


allure.dynamic.link(url: str, link_type: str = LinkType.LINK, name: str = None)

allure.dynamic.issue(url:str,name:str= None)

allure.dynamic.testcase(url: str, name: str = None)

添加与测试用例相关的链接。

allure.attach(body,name=None,attachment_type="text/plain", extension="attach")


allure.attach.file(source,name=None, attachment_type=None, extension=None)

将附件添加到测试报告

@allure.id(id: str)


allure.dynamic.id(id: str)

设置测试用例的id

allure.dynamic.parameter(name,value,excluded=None, mode=None)

显示测试期间使用的参数的name和value。

示例1 添加需求层次结构

@allure.epic("epic")
@allure.feature("features")
@allure.story("assert")
def test_1():
 assert 1 == 1

def test_2():
 allure.dynamic.epic("epic")
 allure.dynamic.feature("features")
 allure.dynamic.story("assert")
 assert 1 == 1

测试用例执行后,生成测试报告如下:点击下面的红框地方可以看到定义的层次结构

示例2 定义套件结构

@allure.parent_suite("ParentSuite")
@allure.suite("Suite")
@allure.sub_suite("Suite-1")
def test_3():
 assert 1 == 1

def test_4():
 allure.dynamic.parent_suite("ParentSuite")
 allure.dynamic.suite("Suite")
 allure.dynamic.sub_suite("Suite-1")
 assert 1 == 1

测试用例执行后,生成测试报告如下:点击下面的红框地方可以看到定义的套件结构

示例 3 定义测试用例标题

class Testcase():
 def test_1(self):
   allure.dynamic.title('测试类中的用例1')
   assert 1 == 1
 
 @allure.title('测试类中的用例2')
 def test_2(self):
   assert 1 == 1

测试用例执行后,生成测试报告如下:点击下面的红框地方可以看到用例的标题(没有定义标题显示的是测试函数名)

示例 4 定义测试用例的步骤

class Testcase():
 def test_1(self):
   allure.dynamic.title('测试类中的用例1')
   with allure.step('先定义变量'):
   		value = 1
   with allure.step('开始断言'):
 		  assert value == 1

 @allure.step('先定义变量')
 def set_value(self):
 		return 1
 
 @allure.step('开始断言')
 def assert_value(self,value):
 		assert value ==1

 @allure.title('测试类中的用例2')
 def test_2(self):
 		value = self.set_value()
 		self.assert_value(value)

分别使用装饰器和上下文管理器的方式设置测试用例的步骤,测试用例执行后,生成测试报告如下:点击下面的红框地方可以看到用例的步骤

示例 5 设置用例的的描述信息

@allure.description('用例断言')
def test_3():
 assert 1 == 1

def test_4():
 allure.dynamic.description('用例断言')
 assert 1 == 1

def test_5():
 """
 断言
 :return: None
 """
 assert 1 == 1

测试用例执行后,生成测试报告如下:点击下面的红框地方可以看到用例的描述

示例 6 设置用例的标记

@allure.tag('tag1','tag2','tag3')
def test_6():
 assert 1 == 1

def test_7():
 allure.dynamic.tag('tag1','tag2','tag3')
 assert 1 == 1

测试用例执行后,生成测试报告如下:点击下面的红框地方可以看到用例的标记信息

示例7 设置用例的严重级别

@allure.severity('critical')
def test_7():
 assert 1 == 1

def test_8():
 allure.dynamic.severity('normal')
 assert 1 == 1

测试用例执行后,生成测试报告如下:下面的红框地方可以看到用例的级别

示例 8 设置用例的链接

@allure.link("https://dev.example.com/", name="Website")
@allure.issue("AUTH-123")
@allure.testcase("TMS-456")
def test_11():
 assert 1 == 1

def test_12():
 allure.dynamic.link("https://dev.example.com/", name="Website")
 allure.dynamic.issue("AUTH-123")
 allure.dynamic.testcase("TMS-456")
 assert 1 == 1

测试用例执行后,生成测试报告如下:下面的红框地方可以看到用例的链接,可以点击跳转。

示例 9 展示用例的参数

def test_15():
 allure.dynamic.parameter('name', 'wang', excluded=None, mode=None)

@pytest.mark.parametrize('name',['wang','li'])
def test_16(name):
 print(name)

用例15使用allure的参数功能,用例16使用pytest的参数功能,测试用例执行后,生成测试报告如下:下面的红框地方可以看到用例的参数信息

示例10 展示用例的附件

从变量中附加内容

allure.attach(body,name=None,attachment_type="text/plain", extension="attach")

在给定的name下添加body作为测试结果的附件(默认为唯一的伪随机字符串)。body必须是bytes或str类型。

def test_17():
  allure.attach('pytest',name="附件",
             attachment_type=allure.attachment_type.TEXT
               )

测试用例执行后,生成测试报告如下:下面的红框地方可以看到用例的附件信息

从文件中阅读附件

allure.attach.file(source,name=None, attachment_type=None, extension=None)

def test_18():
   allure.attach.file('/Users/ue/Desktop/1.png',name="附件",
       attachment_type=allure.attachment_type.PNG
        )

测试用例执行后,生成测试报告如下:下面的红框地方可以看到用例的附件信息(大家可以添加多种格式的附件,示例中的allure.attachment_type有很多种类型)


共勉: 东汉·班固《汉书·枚乘传》:“泰山之管穿石,单极之绠断干。水非石之钻,索非木之锯,渐靡使之然也。”

-----指水滴不断地滴,可以滴穿石头;

-----比喻坚持不懈,集细微的力量也能成就难能的功劳。

----感谢读者的阅读和学习,谢谢大家。

---祝愿大家都能够龙腾虎跃,步步高升!!!

相关推荐

Python入门学习记录之一:变量_python怎么用变量

写这个,主要是对自己学习python知识的一个总结,也是加深自己的印象。变量(英文:variable),也叫标识符。在python中,变量的命名规则有以下三点:>变量名只能包含字母、数字和下划线...

python变量命名规则——来自小白的总结

python是一个动态编译类编程语言,所以程序在运行前不需要如C语言的先行编译动作,因此也只有在程序运行过程中才能发现程序的问题。基于此,python的变量就有一定的命名规范。python作为当前热门...

Python入门学习教程:第 2 章 变量与数据类型

2.1什么是变量?在编程中,变量就像一个存放数据的容器,它可以存储各种信息,并且这些信息可以被读取和修改。想象一下,变量就如同我们生活中的盒子,你可以把东西放进去,也可以随时拿出来看看,甚至可以换成...

绘制学术论文中的“三线表”具体指导

在科研过程中,大家用到最多的可能就是“三线表”。“三线表”,一般主要由三条横线构成,当然在变量名栏里也可以拆分单元格,出现更多的线。更重要的是,“三线表”也是一种数据记录规范,以“三线表”形式记录的数...

Python基础语法知识--变量和数据类型

学习Python中的变量和数据类型至关重要,因为它们构成了Python编程的基石。以下是帮助您了解Python中的变量和数据类型的分步指南:1.变量:变量在Python中用于存储数据值。它们充...

一文搞懂 Python 中的所有标点符号

反引号`无任何作用。传说Python3中它被移除是因为和单引号字符'太相似。波浪号~(按位取反符号)~被称为取反或补码运算符。它放在我们想要取反的对象前面。如果放在一个整数n...

Python变量类型和运算符_python中变量的含义

别再被小名词坑哭了:Python新手常犯的那些隐蔽错误,我用同事的真实bug拆给你看我记得有一次和同事张姐一起追查一个看似随机崩溃的脚本,最后发现罪魁祸首竟然是她把变量命名成了list。说实话...

从零开始:深入剖析 Spring Boot3 中配置文件的加载顺序

在当今的互联网软件开发领域,SpringBoot无疑是最为热门和广泛应用的框架之一。它以其强大的功能、便捷的开发体验,极大地提升了开发效率,成为众多开发者构建Web应用程序的首选。而在Spr...

Python中下划线 ‘_’ 的用法,你知道几种

Python中下划线()是一个有特殊含义和用途的符号,它可以用来表示以下几种情况:1在解释器中,下划线(_)表示上一个表达式的值,可以用来进行快速计算或测试。例如:>>>2+...

解锁Shell编程:变量_shell $变量

引言:开启Shell编程大门Shell作为用户与Linux内核之间的桥梁,为我们提供了强大的命令行交互方式。它不仅能执行简单的文件操作、进程管理,还能通过编写脚本实现复杂的自动化任务。无论是...

一文学会Python的变量命名规则!_python的变量命名有哪些要求

目录1.变量的命名原则3.内置函数尽量不要做变量4.删除变量和垃圾回收机制5.结语1.变量的命名原则①由英文字母、_(下划线)、或中文开头②变量名称只能由英文字母、数字、下画线或中文字所组成。③英文字...

更可靠的Rust-语法篇-区分语句/表达式,略览if/loop/while/for

src/main.rs://函数定义fnadd(a:i32,b:i32)->i32{a+b//末尾表达式}fnmain(){leta:i3...

C++第五课:变量的命名规则_c++中变量的命名规则

变量的命名不是想怎么起就怎么起的,而是有一套固定的规则的。具体规则:1.名字要合法:变量名必须是由字母、数字或下划线组成。例如:a,a1,a_1。2.开头不能是数字。例如:可以a1,但不能起1a。3....

Rust编程-核心篇-不安全编程_rust安全性

Unsafe的必要性Rust的所有权系统和类型系统为我们提供了强大的安全保障,但在某些情况下,我们需要突破这些限制来:与C代码交互实现底层系统编程优化性能关键代码实现某些编译器无法验证的安全操作Rus...

探秘 Python 内存管理:背后的神奇机制

在编程的世界里,内存管理就如同幕后的精密操控者,确保程序的高效运行。Python作为一种广泛使用的编程语言,其内存管理机制既巧妙又复杂,为开发者们提供了便利的同时,也展现了强大的底层控制能力。一、P...