数据驱动应用(parameterized、DDT)
1. parameterized
parameterized是python的一个参数化库,同时支持unittest、nose、pytest单元测试框架
安装:pip install -i https://pypi.tuna.tsinghua.edu.cn/simple requests parameterized
import unittest import time from selenium import webdriver from parameterized import parameterized class TestBaidu(unittest.TestCase):
# 参数化后,要将setup和teardown修改为类方法,不然每个参数被认为一个测试用例,浏览器会随之每次都打开关闭 @classmethod def setUpClass(cls): cls.driver = webdriver.Chrome() cls.base_url = 'http://www.baidu.com' def baidu_search(self, search_key): self.driver.get(self.base_url) self.driver.find_element_by_id('kw').send_keys(search_key) self.driver.find_element_by_id('su').click() time.sleep(3) # 通过parameterized实现参数化 @parameterized.expand([ # 每个元组都被认为是一条测试用例。测试用例中,通过参数来取每个元组中的数据 ('case1', 'selenium'), ('case2', 'unittest'), ('case3', 'parameterized'), ]) def test_search(self, name, search_key): # name对应元组中的第一列数据,search_key对应第二列 self.baidu_search(search_key) self.assertEqual(self.driver.title, search_key + '_百度搜索') @classmethod def tearDownClass(cls): cls.driver.quit() if __name__ == '__main__': unittest.main(verbosity=2) # verbosity设置为2,表示输出更详细的日志
运行结果如下:
test_search_0_case1 (__main__.TestBaidu) ... ok test_search_1_case2 (__main__.TestBaidu) ... ok test_search_2_case3 (__main__.TestBaidu) ... ok ---------------------------------------------------------------------- Ran 3 tests in 15.239s OK
test_search是测试用例的名称,参数化会自动加上0、1、2用来区分每条测试用例,元组中的case1后缀在名称上
2. DDT
DDT(data-driven tests)是针对unittest单元测试框架设计的扩展库。允许使用不同的测试数据来运行一个测试用例,并将其展示为多个测试用例。
安装:pip install -i https://pypi.tuna.tsinghua.edu.cn/simple requests ddt
from selenium import webdriver import unittest import time from ddt import ddt, data, file_data, unpack @ddt # 测试类需要通过@ddt装饰器进行装饰 class TestBaidu(unittest.TestCase): @classmethod def setUpClass(cls): cls.driver = webdriver.Chrome() def search_baidu(self, keyword): self.driver.get('http://www.baidu.com') self.driver.find_element_by_id('kw').send_keys(keyword) self.driver.find_element_by_id('su').click() time.sleep(3) # ddt提供了不同形式的参数化 @data(['case1','selenium'], ['case2', 'python']) # 列表 @unpack def test_baidu1(self, name, keyword): print('第一组测试用例', name) self.search_baidu(keyword) self.assertEqual(self.driver.title, keyword + '_百度搜索') @data(('case1', 'selenium'), ('case2', 'pyton')) # 元组 @unpack def test_baidu2(self, name, keyword): print('第二组测试用例', name) self.search_baidu(keyword) self.assertEqual(self.driver.title, keyword + '_百度搜索') @data({'keyword':'selenium'}, {'keyword':'python'}) # 字典 @unpack def test_baidu3(self, keyword): print('第三组测试用例', keyword) self.search_baidu(keyword) self.assertEqual(self.driver.title, keyword + '_百度搜索') @classmethod def tearDownClass(cls): cls.driver.quit() if __name__ == '__main__': unittest.main(verbosity=2)
DDT同样支持json文件的读取,让我们更关注于数据文件的内容,及在测试用例中的使用。而不需要关心数据文件是如何被读取进来的
keyword.json
使用json文件参数化测试用例
@file_data('./data/keyword.json') # 读取json文件 def test_baidu(self, keyword): self.search_baidu(keyword) self.assertEqual(self.driver.title, keyword + '_百度搜索')
测试结果如下:
test_baidu_00001_case1 (__main__.TestBaidu)
test_baidu_00001_case1 ... ok
test_baidu_00002_case2 (__main__.TestBaidu)
test_baidu_00002_case2 ... ok
test_baidu_00003_case3 (__main__.TestBaidu)
test_baidu_00003_case3 ... FAIL
ps:
参数化中使用全局变量需注意,如果全局变量的赋值和使用在同一个文件中,则不能使用
原因:代码执行顺序
执行时,先走parameterized中的代码,后走业务代码,所以全局变量的值在赋值之前已经获取,为空,会报错,如下123为执行顺序:
class Test_0001(TestBase): def test_1(self): phone = 123 set_assistant_phone_id(phone) # 2 class Test_0011(TestBase): @parameterized.expand([ ('随机手机号', get_assistant_phone_id()) # 1 先执行。还未赋值,所以获取不到数据 ]) def test_1(self, casename, phone): # 3 func1(phone)
解决方案:不使用参数化,将全局变量放在test用例中即可
class Test_0011(TestBase): def test_1(self): func(get_assistant_phone_id()) def test_2(self): func(phone2)
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】凌霞软件回馈社区,博客园 & 1Panel & Halo 联合会员上线
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步