数据驱动---ddt

一、背景

一般进行接口测试时,每个接口的传参都不止一种情况,一般会考虑正向、逆向等多种组合,所以在测试一个接口时,通常会编写多条case,而这些除了传参不同外,并没有什么区别。这个时候就可以利用ddt来管理测试数据,提高代码复用率。

二、ddt安装

ddt是python的第三方库,安装非常简单:

pip install ddt

ddt官方文档地址:https://pypi.org/project/ddt/

三、ddt模块组成

1、ddt模块包含类的装饰器ddt和两个方法装饰器data

  • ddt.ddt:装饰类,也就是继承TestCase的类。

  • ddt.data:装饰测试方法,参数是一系列的值。

  • ddt.file_data:装饰测试方法,参数是文件名。文件可以是json或者yaml类型。

    注意,如果文件是以“.yml”或者".yaml"结尾,ddt会作为yaml类型处理,其他文件都会作为json文件处理。

    如果文件是列表,列表的值会作为测试用例参数,同时,会作为测试用例方法名后缀显示。

    如果文件是字典,字典的key会作为测试用例方法的后缀显示,字典的value会作为测试用例参数。

  • ddt.unpack:传递的是复杂的数据结构时使用,比如使用列表或者元组,添加unpack后,ddt会自动把元组或者列表对应到多个参数上。

2、测试用例方法名生成规则

使用ddt后,会产生一个新的测试用例方法名,即定义的测试用例方法名,比如:def test_one(),这里就是test_one。

data:如果传递过来的数据存在__ name__ 属性,则这里就是该数据的__ name__ 值。如果未定义__ name__ 属性,ddt会尽量将传递过来的值转化为python标识符,作为data显示。比如:(1,3)就转为1_3。需要注意的是,如果数据是字典,则这里就是字典的key。

四、示例

1、使用data传参数给测试用例

from ddt import ddt,data,file_data,unpack
import unittest

@ddt
class MyTestDdt(unittest.TestCase):
    def setUp(self):
        print('start')

    @data(2,4)#使用data传参数给测试用例
    def test_one(self,value):
        print(f'the @data number is :{value}')

    def tearDown(self):
        print('end')

if __name__ == '__main__':
    unittest.main(verbosity=2)

结果:

start
test_one_1_2 (__main__.MyTestDdt) ... ok
the @data number is :2
end
start
the @data number is :4
test_one_2_4 (__main__.MyTestDdt) ... ok
end

----------------------------------------------------------------------
Ran 2 tests in 0.000s

OK

可以看出data分别传参数2和4给测试用例,然后测试用例执行了两遍。另外,注意此时打印出的测试用例名称已经不同,比如:test_one_1_2和test_one_2_4,最后的2和4,分别是传递进去的参数。

2、使用unpack分解list或者tuple

from ddt import ddt,data,file_data,unpack
import unittest

@ddt
class MyTestDdt(unittest.TestCase):
    def setUp(self):
        print('start')

    @data((1,2),(4,5)) #元组
    @unpack #分解
    def test_one(self,value1,value2):
        print(f'the @data number is :{value1,value2}')

    def tearDown(self):
        print('end')

if __name__ == '__main__':
    unittest.main(verbosity=2)

结果:

start
the @data number is :(1, 2)
end
test_one_1__1__2_ (__main__.MyTestDdt) ... ok
start
the @data number is :(4, 5)
test_one_2__4__5_ (__main__.MyTestDdt) ... ok

end
----------------------------------------------------------------------
Ran 2 tests in 0.002s

OK

可以看出data分别传元组参数(1,2)和(4,5)给测试用例,然后测试用例执行了两遍。另外,注意此时打印出的测试用例名称已经不同,比如:test_one_1__ 1__ 2_和test_one_2__ 4__5_ ,最后的1,2和4,5,分别是传递进去的参数。

3、用unpack分解字典

from ddt import ddt,data,file_data,unpack
import unittest

@ddt
class MyTestDdt(unittest.TestCase):
    def setUp(self):
        print('start')

    @data({'value1':1,'value2':2},{'value1':3,'value2':4}) #字典
    @unpack
    def test_one(self,value1,value2):
        print(f'the @data number is :{value1,value2}')

    def tearDown(self):
        print('end')

if __name__ == '__main__':
    unittest.main(verbosity=2)

结果:

test_one_1 (__main__.MyTestDdt) ... ok
start
test_one_2 (__main__.MyTestDdt) ... ok

the @data number is :(1, 2)
----------------------------------------------------------------------
end
Ran 2 tests in 0.000s
start

the @data number is :(3, 4)
OK
end

4、自动化测试实战

这里我们登录新浪邮箱,并利用数据驱动的方式带参进行测试

import unittest
from ddt import ddt,data,unpack
from selenium import webdriver

def testdata():
    '''测试数据'''
    return [('', '', '请输入邮箱名'), ('', '123456', '请输入邮箱名'),
            ('123456', '', '您输入的邮箱名格式不正确')]

@ddt
class MyMailLogin(unittest.TestCase):
    def setUp(self):
        self.url = 'https://mail.sina.com.cn/'
        self.driver = webdriver.Chrome()
        self.driver.maximize_window()
        self.driver.get(self.url)
        self.driver.implicitly_wait(30)

    @data(*testdata()) #这里加*后会将返回数据分为一个个的元组
    @unpack
    def test_login(self,username,password,result):
        '''测试邮箱登录'''
        self.driver.find_element_by_id('freename').clear()
        self.driver.find_element_by_id('freename').send_keys(username)
        self.driver.find_element_by_id('freepassword').clear()
        self.driver.find_element_by_id('freepassword').send_keys(password)
        self.driver.find_element_by_link_text('登录').click()
        freeError = self.driver.find_element_by_xpath('//div[@class="freeError"]//span').text
        self.assertEqual(freeError,result)

    def tearDown(self):
        self.driver.quit()

if __name__ == '__main__':
    unittest.main(verbosity=2)

结果:

test_login_1___________请输入邮箱名__ (__main__.MyMailLogin)
测试邮箱登录 ... ok
test_login_2_______123456____请输入邮箱名__ (__main__.MyMailLogin)
测试邮箱登录 ... ok
test_login_3___123456________您输入的邮箱名格式不正确__ (__main__.MyMailLogin)
测试邮箱登录 ... ok

需要注意@data( * testdata()) 这里加*后会将返回数据分为一个个的元组,否则unpack无法分解数据,就会报错。比如:

def testdata():
    return (('', '', '请输入邮箱名'), ('', '123456', '请输入邮箱名'),
            ('123456', '', '您输入的邮箱名格式不正确'))

print(*testdata()) #三个元组
print(testdata()) #一个元组

结果:

('', '', '请输入邮箱名') ('', '123456', '请输入邮箱名') ('123456', '', '您输入的邮箱名格式不正确')
(('', '', '请输入邮箱名'), ('', '123456', '请输入邮箱名'), ('123456', '', '您输入的邮箱名格式不正确'))

你可能感兴趣的:(数据驱动---ddt)