Unittest框架
unittest 简介
unittest 是python 的Uij界面的单元测试框架,unittest 单元测试提供了创建测试用例,测试套件以及批量执行的方案, unittest 在安装pyhton 以后就直接自带了,直接import unittest 就可以使用。作为单元测试的框架, unittest 也是可以对程序最小模块的一种敏捷化的测试。在自动化测试中,我们虽然不需要做白盒测试,但是必须需要知道所使用语言的单元测试框架。利用单元测试框架,创建一个类,该类继承unittest的TestCase,这样可以把每个case看成是一个最小的单元, 由测试容器组织起来,到时候直接执行,同时引入测试报告。
unittest组成部分及关系
- test ?xture:初始化和清理测试环境,比如创建临时的数据库,文件和目录等,其中setUp()和tearDown()是最常用的方法
- test case:单元测试用例,TestCase是编写单元测试用例最常用的类
- test suite:单元测试用例的集合,TestSuite是最常用的类
- test runner:执行单元测试
- test report:生成测试报告
test case
from selenium import webdriver
from selenium.webdriver.common.by import By
from selenium.webdriver.common.keys import Keys
from selenium.webdriver.support.ui import Select
from selenium.common.exceptions import NoSuchElementException
from selenium.common.exceptions import NoAlertPresentException
import unittest, time, re
class Baidu1(unittest.TestCase):
def setUp(self):
self.driver = webdriver.Firefox()
self.driver.implicitly_wait(30)
self.base_url = "http://www.baidu.com/"
self.verificationErrors = []
self.accept_next_alert = True
def test_baidusearch(self):
driver = self.driver
driver.get(self.base_url + "/")
driver.find_element_by_id("kw").click()
driver.find_element_by_id("kw").clear()
driver.find_element_by_id("kw").send_keys(u"测试")
driver.find_element_by_id("su").click()
def test_hao(self):
driver = self.driver
driver.get(self.base_url + "/")
driver.find_element_by_link_text("hao123").click()
self.assertEqual(u"hao123_上网从这里开始", driver.title)
def tearDown(self):
self.driver.quit()
self.assertEqual([], self.verificationErrors)
if __name__ == "__main__":
unittest.main(verbosity=2)
构建测试套件 完整的单元测试很少只执行一个测试用例,开发人员通常都需要编写多个测试用例才能对某一软件功能进行比较完整的测试,这些相关的测试用例称为一个测试用例集,在unittest中是用TestSuite 类来表示的。
addTest() 的应用 当有多个或者几百测试用例的时候, 这样就需要一个测试容器( 测试套件) ,把测试用例放在该容器中进行执行,unittest 模块中提供了TestSuite 类来生成测试套件,使用该类的构造函数可以生成一个测试套件的实例,该类提供了addTest来把每个测试用例加入到测试套件中。只能一个一个的添加。
import unittest,csv
import os,sys
import time
import testbaidu1
import testbaidu2
def createsuite():
suite = unittest.TestSuite()
suite.addTest(testbaidu1.Baidu1("test_baidusearch"))
suite.addTest(testbaidu2.Baidu2("test_baidusearch"))
return suite
if __name__=="__main__":
suite=createsuite()
runner = unittest.TextTestRunner(verbosity=2)
runner.run(suite)
makeSuite()和TestLoader()的应用 在unittest 框架中提供了makeSuite() 的方法,makeSuite可以实现把测试用例类内所有的测试case组成的测试套件TestSuite ,unittest 调用makeSuite的时候,只需要把测试类名称传入即可。TestLoader 用于创建类和模块的测试套件,一般的情况下,使TestLoader().loadTestsFromTestCase(TestClass)来加载测试类。
def createsuite():
suite = unittest.TestSuite()
suite.addTest(unittest.makeSuite(testbaidu1.Baidu1))
suite.addTest(unittest.makeSuite(testbaidu2.Baidu2))
return suite
'''
suite1 = unittest.TestLoader().loadTestsFromTestCase(testbaidu1.Baidu1)
suite2 = unittest.TestLoader().loadTestsFromTestCase(testbaidu2.Baidu2)
suite = unittest.TestSuite([suite1, suite2])
return suite
discover()的应用 discover 是通过递归的方式到其子目录中从指定的目录开始, 找到所有测试模块并返回一个包含它们对象的TestSuite ,然后进行加载与模式匹配唯一的测试文件,discover 参数分别为discover(dir,pattern,top_level_dir=None)
import unittest,csv
import os,sys
import time
def createsuite():
discover=unittest.defaultTestLoader.discover('../test',pattern='test*.py',top_level_dir=None)
print discover
return discover
if __name__=="__main__":
suite=createsuite()
runner = unittest.TextTestRunner(verbosity=2)
runner.run(suite)
用例的执行顺序 unittest 框架默认加载测试用例的顺序是根据ASCII 码的顺序,数字与字母的顺序为: 09,AZ,a~z 。所以, TestAdd 类会优先于TestBdd 类被发现, test_aaa() 方法会优先于test_ccc() 被执行。对于测试目录与测试文件来说, unittest 框架同样是按照这个规则来加载测试用例。addTest()方法按照增加顺序来执行。 忽略用例执行 放在方法前
@unittest.skip("skipping")
def test_baidusearch(self):
unittest断言
自动化的测试中, 对于每个单独的case来说,一个case的执行结果中, 必然会有期望结果与实际结果, 来判断该case是通过还是失败, 在unittest 的库中提供了大量的实用方法来检查预期值与实际值, 来验证case的结果, 一般来说, 检查条件大体分为等价性, 逻辑比较以及其他, 如果给定的断言通过, 测试会继续执行到下一行的代码, 如果断言失败, 对应的case测试会立即停止或者生成错误信息( 一般打印错误信息即可) ,但是不要影响其他的case执行。
self.assertEqual("admin", driver.find_element_by_link_text("admin").text)
HTML报告生成
脚本执行完毕之后,还需要看到HTML报告,下面我们就通过HTMLTestRunner.py 来生成测试报告。HTMLTestRunner支持python2.7。
import unittest,csv
import os,sys
import time
import HTMLTestRunner
def createsuite():
discover=unittest.defaultTestLoader.discover('../test',pattern='test*.py',top_level_dir=None)
print discover
return discover
if __name__=="__main__":
curpath=sys.path[0]
now=time.strftime("%Y-%m-%d-%H %M %S",time.localtime(time.time()))
if not os.path.exists(curpath+'/resultreport'):
os.makedirs(curpath+'/resultreport')
filename=curpath+'/resultreport/'+now+'resultreport.html'
with open(filename,'wb') as fp:
runner=HTMLTestRunner.HTMLTestRunner(stream=fp,title=u'测试报告',description=u'用例执行情况',verbosity=2)
suite=createsuite()
runner.run(suite)
异常捕捉与错误截图
用例不可能每一次运行都成功,肯定运行时候有不成功的时候。如果可以捕捉到错误,并且把错误截图保存,这将是一个非常棒的功能,也会给我们错误定位带来方便。
def savescreenshot(self,driver,file_name):
if not os.path.exists('./image'):
os.makedirs('./image')
now=time.strftime("%Y%m%d-%H%M%S",time.localtime(time.time()))
driver.get_screenshot_as_file('./image/'+now+'-'+file_name)
time.sleep(1)
数据驱动
python 的unittest 没有自带数据驱动功能。所以如果使用unittest,同时又想使用数据驱动,那么就可以使用DDT来完成。
@data(“1”,“2”,“3”)适合传递一个参数的,并把方法里的量改为变量,数据量大时,可以通过file_data加JSON文件的方式来,例如下面的例子会一次查找 :“王凯”, “Lisa”, “特朗普”, "蒋欣"等四个数据
from selenium import webdriver
import unittest
import time
from ddt import ddt, unpack, data, file_data
import sys, csv
@ddt
class Baidu1(unittest.TestCase):
def setUp(self):
self.driver = webdriver.Chrome()
self.driver.implicitly_wait(30)
self.base_url = "http://www.baidu.com/"
self.driver.maximize_window()
self.verificationErrors=[]
self.accept_next_alert = True
def tearDown(self):
self.driver.quit()
self.assertEqual([],self.verificationErrors)
@data("王凯", "Lisa", "特朗普", "蒋欣")
def test_baidu1(self, value):
driver = self.driver
driver.get(self.base_url + "/")
driver.maximize_window()
driver.find_element_by_id("kw").clear()
driver.find_element_by_id("kw").send_keys(value)
driver.find_element_by_id("su").click()
time.sleep(3)
当@data需要传递的是多个参数的时候 要用一个中括号将一组数据抱起来,并加上 @unpack(表示解绑),例如: @data([“1”,“1”,“1”,“1”],[“1”,“1”,“1”,“2”],[“1”,“2”,“1”,“1”]) @unpack 具体代码
@data(["1","1","1","1"],["1","1","1","2"],["1","2","1","1"])
@unpack
def test_triangle1(self,value1,value2,value3,value4):
driver = self.driver
driver.find_element_by_id("username").send_keys(value1)
time.sleep(1)
driver.find_element_by_id("idcard").send_keys(value2)
time.sleep(1)
driver.find_element_by_id("pwd").send_keys(value3)
time.sleep(1)
driver.find_element_by_id("pwd2").send_keys(value4)
time.sleep(1)
还可以写一个JSON文件,读取JSON文件
@file_data("test.json")
def test_triangle1(self,value1,value2,value3,value4):
driver = self.driver
driver.find_element_by_id("username").send_keys(value1)
time.sleep(1)
driver.find_element_by_id("idcard").send_keys(value2)
time.sleep(1)
driver.find_element_by_id("pwd").send_keys(value3)
time.sleep(1)
driver.find_element_by_id("pwd2").send_keys(value4)
很多时候测试的数据比较多,这样就不可能将成千上万的数据放在代码中,这样就会显得代码很臃肿,所以就需要将这些测试数据和代码进行分离,需要数据的时候直接在文件中读取就可以了,这样不仅显得代码简洁美观,还易于理解并添加数据。
import csv,os,sys,time
from selenium import webdriver
import unittest
from ddt import ddt, unpack, data
def getCsv(file_name):
rows = []
path = sys.path[0]
print(path)
with open(path + "/data/" + file_name, 'rt') as f:
readers = csv.reader(f, delimiter=',', quotechar='|')
next(readers, None)
for row in readers:
temprows = []
for i in row:
temprows.append(i)
rows.append(temprows)
return rows
@ddt
class test01(unittest.TestCase):
def setUp(self):
self.driver = webdriver.Chrome()
self.url = "http://localhost:8080/dd/reg.html"
self.driver.get(self.url)
self.driver.maximize_window()
def tearDown(self):
self.driver.quit()
@data(*getCsv("wenjianqudong01.txt"))
@unpack
def test_triangle1(self, value1, value2, value3, value4):
driver = self.driver
driver.find_element_by_id("username").send_keys(value1)
driver.find_element_by_id("idcard").send_keys(value2)
driver.find_element_by_id("pwd").send_keys(value3)
driver.find_element_by_id("pwd2").send_keys(value4)
driver.find_element_by_css_selector("#button1 > div > input[type=button]:nth-child(1)").click()
time.sleep(1)
alert = driver.switch_to.alert
print(value1 + value2 + value3 + value4 + ":" + alert.text)
alert.accept()
|