WebUI之数据驱动与Unittest(一)

一、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文件


    image.png

    image.png

    image.png

附录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


image.png

    后续继续更新添加断言,以及更加系统、更加实用的方法。以上内容仅能学习使用,不完善,在工作中使用不方便,而且架构也不好。

三、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)
结果:
['孔子', '孟子', '庄子']
['老子', '孙武', '静静']



image.png

方法五:

@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)

结果:

image.png

说明:
    上面的这种方法用于多数据的情况,如注册用户测试用例,不会造成用例文档很长,使界面变的更加简洁:

csv格式:
acount,result
[用户名,密码],pass



学习之路漫漫,愿你我一路共勉!

你可能感兴趣的:(WebUI之数据驱动与Unittest(一))