最近工作中向同事学习了利用Python的unittest做自动化测试,另外简单查些资料做些补充。
测试的通用规则
测试单元应该集中于小部分的功能,并且证明它是对的。
每个测试单元应该完全独立,它们都能够单独运行,也可以在测试套件中运行,而不用考虑被调用的顺序。
unittest
unittest 是Python内置的单元测试测试框架,具备:
编写用例,即 test fixture
组织用例,即 test case,test suite
执行用例,即 test Loader
输出报告,即 test runner
等自动化测试框架必备条件。
test fixture
一个测试用例的初始化准备及环境还原,主要是 setUp() 和 setDown() 方法。
test case
一个完整的测试单元,执行该测试单元可以完成对某一个问题的验证。完整体现在:
测试前环境准备
执行测试代码
测试后环境还原
test suite
多个测试用例的集合,测试套件或测试计划。
test Loader
加载 TestCase 到 TestSuite 中的。
test runner
执行测试用例,并将测试结果保存到TextTestResult实例中,包括运行了多少测试用例, 成功了多少,失败了多少等信息。
unittest的工作原理
// todo
unittest实战
待测模块
def is_prime(number):
if number < 0 or number in (0, 1):
return False;
for element in range(2, number):
if number % element == 0:
return False;
return True
def add(a, b):
return a + b;
def divide(a, b):
return a / b;
使用unittest编写测试用例
import unittest
class TestMyFunc(unittest.TestCase):
def setup(self):
print("每个用例执行前后会调用setUp方法准备环境")
def tearDown(self):
printf("每个用例执行前后会调用tearDown方法进行环境清理");
def test_is_prime(self):
'''test method is_prime'''
self.assertEqual(is_prime(5))
self.assertEqual(is_prime(0))
self.assertEqual(is_prime(-1))
def test_add(self)
'''test method divide'''
self.assertEqual(3, add(1, 2))
self.assertEqual(3, add(2, 2))
def test_divide(self):
'''test method divide'''
self.assertEqual(3, divide(6, 3))
self.assertEqual(3, divide(5, 2))
if __name == '__main__':
unittest.main()
框架解决自动化需求的4个问题
如何控制用例执行顺序
在unittest中,用例是以test开头的方法定义的,默认执行顺序是根据用例名称升序进行,为不是用例定义的先后顺序。在unittest中解决顺序的问题是使用TestSuite。代码如下:
if __name__ == '__main__':
tests = [TestMyFunc("test_is_prime"), TestMyFunc("test_add"), TestMyFunc("test_divide")]
suite = unittest.TestSuite()
suite.addTest(tests);
runner = unittest.TextTestRunner()
runner.run(suite)
如何让多个用例共用setup,teardown
unitest 的 setup,teardown 会在用例执行前后执行一次,如上面测试用例类中有3个测试用例,那么每个用例执行前后执行 setup,执行后会执行teardown,即 setup,teardown 总过调用三次。但考虑实际自动化测试场景,多个用例只需执行一次setup,全部用例执行完成后,执行一次teardown,针对该种场景,unittest的处理方法是使用setupclass,teardownclass,注意如下:
class TestMyFunc(unittest.TestCase):
@classmethod
def setUpClass(cls):
print('所有用例执行前会调用一次setUp准备环境')
@classmethod
def tearDownClass(cls):
print('所有用例执行后会调用一次tearDown进行环境清理')
def test_is_prime(self):
''' test method divide'''
self.assertEqual(2, divide(6, 3))
slef.assertNotEqual(2, divide(5, 2))
if __name__ == '__main__':
unittest.main()
如何跳过用例
在自动化测试中,可能会用到选择性执行用例的情况,在unittest中解决的方法是使用skip装饰器,其中skip装饰器主要有3种:unittest.skip(reason),unittest.skipif(condition, reason),unittest.skipUnless(condition, reason),即在满足condition条件下跳过改用例,reason用于描述跳过的原因。
...
@unittest.skipUnless(sys.platform.startwith('linux'), 'requires Linux')
def test_devide(self):
self.assertEqual(2, divide(6, 3))
self.assertEqual(2, divide(5, 2))
if __name__=='__main__':
unittest.main()
如何生成自动化测试报告
unittest中模块生成的报告格式为txt,如果想生成html格式的报告,可以使用HtmlTestRunner模块,安装后导入该模块,使用HTMLTestRunner代替默认的TextTestRunner()执行测试用例即可。
from HtmlTestRunner import HTMLTestRunner
suite = unittest.TestSuite()
suite.addTest(unittest.TestLoader().loadTestFromTestCase(TestMyfunc))
sunner.run(suite)