unittest是python自带的一个库,不需要安装
在调用的时候
记得py文件要以test开头
py文件中写的类也要以Test开头,并且要继承(unittest.TestCase)
下面为一个简单的例子。
- 文件名及类都要以test开头。
- 运行的话,不要直接用python运行,
最简单的测试脚本
import unittest
class TestLogin(unittest.TestCase):
def test_login(self):
try:
self.assertEqual(1,1)
# 1/0
except AssertionError as e:
print('断言失败',e)
if __name__ == '__main__':
print("python running")
unittest.main()
运行的方法
用pycharm来执行的话,这样使用,我们鼠标如果在类上停下来,那么我们就会只执行这个类下面的函数,如果在某一个函数下运行,那么我们就只会执行这个函数。
python -m unittest 路径的方法来运行(也可以加模块名或者类名的方式)
多个测试用例的执行顺序
测试用例的执行顺序是按照按ASCII码进行编码的
如果我们想让用例按照顺序执行,那么我可以在后面加上数字
例如:
def test_login_1_success(self):
pass
def test_login_2_error(self):
pass
上面的这段代码,在执行测试用例时,是一定会先执行第一个,然后是第二个,这个就是加了数字的好处。
断言
说起个常用的断言吧,其余的大家可以自行去看官方文档。
前置条件 setup()
每个测试用例方法执行之前都会运行的代码,也就是说你写了几个test方法,就会执行几次
-
teader down的话是后置的一个函数
也就是每次用例执行完,都会执行的操作。 -
一个简单的小例子
import unittest
class TestLogin(unittest.TestCase):
def setUp(self) -> None:
print("测试用例开始执行")
a = 1
b = 2
# print(a,b)
def tearDown(self) -> None:
print("测试用例执行完毕")
def test_login_0(self):
try:
self.assertEqual(a,b)
# 1/0
except AssertionError as e:
print('断言失败',e)
# raise e
def test_login_1(self):
try:
self.assertEqual(a,b)
# 1/0
except AssertionError as e:
print('断言失败',e)
# raise e
#
if __name__ == '__main__':
# print("python running")
unittest.main()
我们用python来运行一下
C:\test_1>python test_login.py
结果如下:
C:\test_1>python test_login.py
测试用例开始执行
测试用例执行完毕
E测试用例开始执行
测试用例执行完毕
E
======================================================================
ERROR: test_login_0 (__main__.TestLogin)
----------------------------------------------------------------------
Traceback (most recent call last):
File "test_login.py", line 24, in test_login_0
self.assertEqual(a,b)
NameError: name 'a' is not defined
======================================================================
ERROR: test_login_1 (__main__.TestLogin)
----------------------------------------------------------------------
Traceback (most recent call last):
File "test_login.py", line 31, in test_login_1
self.assertEqual(a,b)
NameError: name 'a' is not defined
----------------------------------------------------------------------
Ran 2 tests in 0.001s
FAILED (errors=2)
测试套件及加载器
收集测试用例:TestLoader,加载器,加载测试用例
放到测试集合(测试套件)TestSuite
我们拿例子说话
import os
import unittest
# 1,初始化 testloader
testloader = unittest.TestLoader()
# 2,查找测试用例,加载
# testloader.discover(文件夹的路径,‘demo.py’)
# 第一个参数的含义就是输入正确的文件夹路径,我使用dir_path 输出的是一个绝对路径
# 第二个参数的含义就是在这个路径下查找以什么开头的py文件,当然不写话,默认是‘test*.py’
#3,如果想要运行就放到特定的文件夹之中,也就是上面的suite
dir_path = os.path.dirname(os.path.abspath(__file__))
case_path = os.path.join(dir_path)
suite = testloader.discover(case_path)
print(suite)
这是那个文件夹中的测试文件,看看那些能够进入testsuite中
C:\Users\Admin\AppData\Local\Microsoft\WindowsApps\python.exe C:/test_1/run_test.py
<unittest.suite.TestSuite tests=[<unittest.suite.TestSuite tests=[<unittest.suite.TestSuite tests=[<TestAssert.TestLogin testMethod=test_login_0>]>]>, <unittest.suite.TestSuite tests=[<unittest.suite.TestSuite tests=[<test_login.TestLogin testMethod=test_login_0>, <test_login.TestLogin testMethod=test_login_1>]>]>]>
Process finished with exit code 0
运行器–测试报告
主要使用的是TextTestRunner,如何使用我都放到下面的代码中注释中了
# 4,生成报告
# 我们的报告重新放在一个report文件夹中
# 我们使用TextTestRunner来执行测试报告
report_path = os.path.join(dir_path,"report")
if not os.path.exists(report_path):
os.mkdir(report_path)
file_path = os.path.join(report_path,'test_reslut.txt')
with open(file_path,'w',encoding='utf-8') as f:
# 初始化运行初期,是以普通文本生成测试报告 TextTestRunner
runner = unittest.TextTestRunner(f)
runner.run(suite)
运行的报告如下
.EE
======================================================================
ERROR: test_login_0 (test_login.TestLogin)
----------------------------------------------------------------------
Traceback (most recent call last):
File "C:\test_1\test_login.py", line 24, in test_login_0
self.assertEqual(a,b)
NameError: name 'a' is not defined
======================================================================
ERROR: test_login_1 (test_login.TestLogin)
----------------------------------------------------------------------
Traceback (most recent call last):
File "C:\test_1\test_login.py", line 31, in test_login_1
self.assertEqual(a,b)
NameError: name 'a' is not defined
----------------------------------------------------------------------
Ran 3 tests in 0.001s
FAILED (errors=2)
这个测试报告最上,一个点代表的是通过,E代表的是出现异常,但是不是断言引起的,F的话就代表,用例中断言,出错
- TextTestRunner的参数
verbosity 这个参数代表的报告的详细程度
总结
额外补充
加载指定的测试用例py
- 这个记得一定到将你所使用py文件导入进来
- 导入有一个快捷见
alt+shfit+enter
- 当然也可以指标悬停,找到对于的选项导入
import TestAssert import test_login
import os
import unittest
# 1,初始化 testloader
import TestAssert
import test_login
testloader = unittest.TestLoader()
# 2,查找测试用例,加载
# testloader.discover(文件夹的路径,‘demo.py’)
# 第一个参数的含义就是输入正确的文件夹路径,我使用dir_path 输出的是一个绝对路径
# 第二个参数的含义就是在这个路径下查找以什么开头的py文件,当然不写话,默认是‘test*.py’
#3,如果想要运行就放到特定的文件夹之中,也就是上面的suite
dir_path = os.path.dirname(os.path.abspath(__file__))
case_path = os.path.join(dir_path)
suite = testloader.loadTestsFromModule(test_login)
suite2 = testloader.loadTestsFromModule(TestAssert)
# 将这两个测试套件合并添加到一个总的测试套件
suite_totoal = unittest.TestSuite()
suite_totoal.addTest(suite)
suite_totoal.addTest(suite2)
print(suite_totoal)
# 4,生成报告
# 我们的报告重新放在一个report文件夹中
# 我们使用TextTestRunner来执行测试报告
report_path = os.path.join(dir_path,"report")
if not os.path.exists(report_path):
os.mkdir(report_path)
file_path = os.path.join(report_path,'test_reslut.txt')
with open(file_path,'w',encoding='utf-8') as f:
# 初始化运行初期,是以普通文本生成测试报告 TextTestRunner
runner = unittest.TextTestRunner(f,verbosity= 1)
runner.run(suite_totoal)
加载指定的测试用例的类
- 道理一样,只是做了如下修改
suite = testloader.loadTestsFromTestCase(test_login.TestLogin)
# 将这两个测试套件合并添加到一个总的测试套件
suite_totoal = unittest.TestSuite()
suite_totoal.addTest(suite)
# suite_totoal.addTest(suite2)
print(suite_totoal)
网页版的测试报告文件生成
我们看一下部分的源码,理解一下其中的内容
class HTMLTestRunner(Template_mixin):
"""
"""
def __init__(self, stream=sys.stdout, verbosity=2,title=None,description=None,tester=None):
#verbosity=2 日志等级,2最详细
title=None 测试的标题
description 对测试的描述
tester 测试的人员
整体的代码放到下面,大家可以自己运行一下
import os
import unittest
from HTMLTestRunnerNew import HTMLTestRunner
# 1,初始化 testloader
testloader = unittest.TestLoader()
# 2,查找测试用例,加载
# testloader.discover(文件夹的路径,‘demo.py’)
# 第一个参数的含义就是输入正确的文件夹路径,我使用dir_path 输出的是一个绝对路径
# 第二个参数的含义就是在这个路径下查找以什么开头的py文件,当然不写话,默认是‘test*.py’
#3,如果想要运行就放到特定的文件夹之中,也就是上面的suite
dir_path = os.path.dirname(os.path.abspath(__file__))
case_path = os.path.join(dir_path)
suite = testloader.discover(case_path)
print(suite)
# 4,生成报告
# 我们的报告重新放在一个report文件夹中
# 我们使用TextTestRunner来执行测试报告
report_path = os.path.join(dir_path,"report")
if not os.path.exists(report_path):
os.mkdir(report_path)
file_path = os.path.join(report_path,'test_reslut.htm')
# 网页版的一定要使用二进制的方式打开
with open(file_path,'wb') as f:
# 初始化运行初期,是以普通文本生成测试报告 TextTestRunner
# runner = unittest.TextTestRunner(f,verbosity= 1)
runner = HTMLTestRunner(f,
title="好哈学习天天向上",
description="本次为unnittest框架的学习",
tester="chocolate"
)
runner.run(suite)
结果展示效果如下:
- 我们在继续扩展一下,现在的测试报告是已经写死到系统中的,我们现在想把时间戳加进去,
- 其实工作中是很实用的
直接放相应的代码
from datetime import datetime
ts = datetime.now().strftime('%Y-%m-%d-%H-%M-%S')
file_name = 'test_result_{}.htm'.format(ts)
file_path = os.path.join(report_path,file_name)
# 网页版的一定要使用二进制的方式打开
with open(file_path,'wb') as f:
# 初始化运行初期,是以普通文本生成测试报告 TextTestRunner
# runner = unittest.TextTestRunner(f,verbosity= 1)
runner = HTMLTestRunner(f,
title="好哈学习天天向上",
description="本次为unnittest框架的学习",
tester="chocolate"
)
runner.run(suite)
总结
工作中我们经常使用的一些点大概都涉及到了,剩下就是做接口测试的时候,我们将相应的一个业务逻辑关系处理好就可以了