一、Unittest框架
下面是最简单的使用Unittest框架的方法,所有的内容都放在里面,但是不利于后期维护。
该使用了全局变量,在这种只有一个测试集与用例的场合没有问题,但是一个文件内有多个用例集与多个用例就会产生问题,要么使用在打开的浏览器器中,再打开一个页面执行其他用例,要么使用其他方法。
from selenium import webdriver
import unittest
#使用unittest必须先使用class定义一个类,且以Test开头,并继承unittest.TestCase
class Testbaidu_search(unittest.TestCase):
# 必须使用@classmethod装饰器,这是使用tearDownClass与setUpClass的基本要求
#setUpClass表示下面的代码Testbaidu_search测试集只会在开始时运行一次
@classmethod
def setUpClass(cls) -> None:
global driver
driver=webdriver.Chrome()
driver.get('https://www.baidu.com/')
def test_search(self):
driver.find_element_by_id('kw').send_keys('孔子')
driver.find_element_by_id('su').click()
@classmethod
# tearDownClass表示下面的代码Testbaidu_search测试集只会在结束时运行一次
def tearDownClass(cls) -> None:
driver.quit()
if __name__ == '__main__':
unittest.main()
改进:
from selenium import webdriver
import time
import unittest
#使用unittest必须先使用class定义一个类,且以Test开头,并继承unittest.TestCase
class Testbaidu_search(unittest.TestCase):
# 必须使用@classmethod装饰器,这是使用tearDownClass与setUpClass的基本要求
#setUpClass表示下面的代码Testbaidu_search测试集只会在开始时运行一次
@classmethod
def setUpClass(cls) -> None:
#由于setUpClass(cls)是一个类的方法,内部的变量可以被其他方法通过self.变量名访问,而cls,只是代表setUpClass本身,与self是一样的意思,在其他方法内通过self调用访问即可。
cls.driver=webdriver.Chrome()
cls.driver.get('https://www.baidu.com/')
def test_search(self):
self.driver.find_element_by_id('kw').send_keys('孔子')
self.driver.find_element_by_id('su').click()
@classmethod
# tearDownClass表示下面的代码Testbaidu_search测试集只会在结束时运行一次
def tearDownClass(cls) -> None:
time.sleep(3)
cls.driver.quit()
if __name__ == '__main__':
unittest.main()
二、ddt数据驱动
打开ddt模块的源代码,发现使用ddt里面的方法需要在Unittest的testcase中间使用,使用情形为装饰器样式。下面为常用方法:
- ddt 放在定义的测试集前面,表明这个测试集启用数据驱动
- data(数据内容或者*数据变量) 放在测试方法前面,表明这个方法使用的是什么数据
- unpack 对data的数据进行解包
-
file_data(value, yaml_loader=None) 可以直接读取yaml和json文件
附录ddt数据驱动与Unittest结合的代码:
from selenium import webdriver
from unittest import TestCase
import unittest
from ddt import ddt,data,unpack
import time
search_data=(['孔子'],['孟子'],['庄子'])
#使用装饰器,表明该测试集采用ddt驱动
@ddt
#使用unittest必须先使用class定义一个类,且以Test开头,并继承unittest.TestCase
class Testbaidu_search(unittest.TestCase):
# 必须使用@classmethod装饰器,这是使用tearDownClass与setUpClass的基本要求
#setUpClass表示下面的代码Testbaidu_search测试集只会在开始时运行一次
@classmethod
def setUpClass(cls) -> None:
global driver
driver=webdriver.Chrome()
driver.get('https://www.baidu.com/')
#data取数据
# @data(['孔子'],['孟子'],['庄子'])
#如果传入的是一个变量接收的数据集,那么需要在变量前面加*
@data(*search_data)
#unpack解压数据包
@unpack
#由于只有一个变量,在测试用例用中,加入一个变量名,用于接收解压到的数据,针对多个的情况后面再详细说明
def test_search(self,value):
driver.find_element_by_id('kw').send_keys(value)
driver.find_element_by_id('su').click()
time.sleep(3)
driver.find_element_by_id('kw').clear()
#上面的用例思路可以用在需要反复输入测试的场合
@classmethod
# tearDownClass表示下面的代码Testbaidu_search测试集只会在结束时运行一次
def tearDownClass(cls) -> None:
driver.quit()
if __name__ == '__main__':
unittest.main(verbosity=2)
下图可以看到,完成了三个测试用例,并且用例名自动加了1
后续继续更新添加断言,以及更加系统、更加实用的方法。以上内容仅能学习使用,不完善,在工作中使用不方便,而且架构也不好。
三、ddt中data与unpack的详细讲解
这里只中间部分代码,即存在区别的地方:
方法一:将括号内的每个子列表视为一个数据,在Unittest进行反复调用,执行用例
@data(['孔子'],['孟子'],['庄子'])
方法二:data里面存在一个列表(列表内多个数据),需要使用unpack解一次包
@data(['孔子', '孟子', '庄子'])
@unpack
方法三:变量接收的数据需要需要此格式解包:@data(*变量名)
#我定义了一个读取csv文件的函数,得到的结果是一个列表,这个函数在后续的(二)中讲解
#以下代码等同于直接赋值:search_data=['孔子','孟子','庄子']
search_data=read_csv('CVS数据表.csv',col_name='name')
#函数返回:['孔子','孟子','庄子']
#解包
@data(*search_data)
方法四:多重解包
下面代码会列表视为一个数据进行迭代
import unittest
from ddt import ddt,data,unpack
@ddt
#使用unittest必须先使用class定义一个类,且以Test开头,并继承unittest.TestCase
class Testbaidu_search(unittest.TestCase):
def setUp(self) -> None:
pass
data_file=(['孔子', '孟子', '庄子'],['老子', '孙武', '静静'])
@data(*data_file)
def test_jiebao(self,value):
print(value)
def tearDown(self) -> None:
pass
if __name__ == '__main__':
unittest.main(verbosity=2)
结果:
['孔子', '孟子', '庄子']
['老子', '孙武', '静静']
方法五:
@ddt
#使用unittest必须先使用class定义一个类,且以Test开头,并继承unittest.TestCase
class Testbaidu_search(unittest.TestCase):
def setUp(self) -> None:
pass
data_file=(['孔子', '孟子', '庄子'],['老子', '孙武', '静静'])
@data(*data_file)
@unpack
def test_jiebao(self,name1,name2,name3):
print(name1)
print(name2)
print(name3)
def tearDown(self) -> None:
pass
if __name__ == '__main__':
unittest.main(verbosity=2)
结果:
说明:
上面的这种方法用于多数据的情况,如注册用户测试用例,不会造成用例文档很长,使界面变的更加简洁:
csv格式:
acount,result
[用户名,密码],pass
学习之路漫漫,愿你我一路共勉!