Python 數據驅動工具:DDT

背景

python 的unittest 沒有自帶數據驅動功能。python

因此若是使用unittest,同時又想使用數據驅動,那麼就能夠使用DDT來完成。json

DDT是 「Data-Driven Tests」的縮寫。數據結構

資料:http://ddt.readthedocs.io/en/latest/post

使用方法

dd.ddt:

裝飾類,也就是繼承自TestCase的類。測試

ddt.data:

裝飾測試方法。參數是一系列的值。spa

ddt.file_data:

裝飾測試方法。參數是文件名。文件能夠是json 或者 yaml類型。code

注意,若是文件以」.yml」或者」.yaml」結尾,ddt會做爲yaml類型處理,其餘全部文件都會做爲json文件處理。blog

若是文件中是列表,每一個列表的值會做爲測試用例參數,同時做爲測試用例方法名後綴顯示。繼承

若是文件中是字典,字典的key會做爲測試用例方法的後綴顯示,字典的值會做爲測試用例參數。three

ddt.unpack:

傳遞的是複雜的數據結構時使用。好比使用元組或者列表,添加unpack以後,ddt會自動把元組或者列表對應到多個參數上。字典也能夠這樣處理。參見下面的示例2.

測試用例方法名生成規則

使用ddt後,會產生一個新的測試用例方法名:以前的測試用例方法名_ordinal_data

以前的測試用例方法名:即定義的測試用例方法名。好比def test_large(),這裏就是test_large

ordinal:整數,從1開始遞加。

data:若是傳遞過來的數據存在__name__屬性,則這裏就是該數據的__name__值。若是未定義__name__屬性,ddt會盡可能將傳遞過來的數據轉化爲python標識符,做爲data顯示。好比(3,2)就轉化爲3_2。須要注意的是,若是數據是字典,則這裏就是字典的key。

使用示例

1. data直接放入數值

須要導入ddt包,而後再TestCase類上採用@ddt進行裝飾,測試方法上裝飾@data()。

data能夠是數值,也能夠是字符串。

複製代碼
import unittest
from ddt import ddt, data
from ddt_demo.mycode import larger_than_two

@ddt
class FooTestCase(unittest.TestCase):
    
    @data(3, 4, 12, 23)
    def test_larger_than_two(self, value):
        self.assertTrue(larger_than_two(value))

    @data(1, -3, 2, 0)
    def test_not_larger_than_two(self, value):
        self.assertFalse(larger_than_two(value))

    @data(u'ascii', u'non-ascii-\N{SNOWMAN}')
    def test_unicode(self, value):
        self.assertIn(value, (u'ascii', u'non-ascii-\N{SNOWMAN}'))
        
if __name__=='__main__':
    unittest.main(verbosity=2)
複製代碼

輸出以下:

複製代碼
test_larger_than_two_1_3 (__main__.FooTestCase) ... ok
test_larger_than_two_2_4 (__main__.FooTestCase) ... ok
test_larger_than_two_3_12 (__main__.FooTestCase) ... ok
test_larger_than_two_4_23 (__main__.FooTestCase) ... ok
test_not_larger_than_two_1_1 (__main__.FooTestCase) ... ok
test_not_larger_than_two_2__3 (__main__.FooTestCase) ... ok
test_not_larger_than_two_3_2 (__main__.FooTestCase) ... ok
test_not_larger_than_two_4_0 (__main__.FooTestCase) ... ok
test_unicode_1_ascii (__main__.FooTestCase) ... ok
test_unicode_2_non_ascii__ (__main__.FooTestCase) ... ok

----------------------------------------------------------------------
Ran 10 tests in 0.001s

OK
複製代碼

能夠看到上面只寫了3個測試方法,可是最後run了10個用例。

這裏測試方法後會被ddt加一個後綴,ddt會嘗試把測試數據轉化爲後綴附在測試方法後,組成一個新的名字。

2. data放入複雜的數據結構

使用複雜的數據結構時,須要用到@unpack,同時測試方法的參數須要使用對應的多個,好比下面的frist_value 以及 second_value。

複製代碼
import unittest
from ddt import ddt, data,unpack

@ddt
class FooTestCase(unittest.TestCase):

    @data((3, 2), (4, 3), (5, 3))
    @unpack
    def test_tuples_extracted_into_arguments(self, first_value, second_value):
        self.assertTrue(first_value > second_value)

    @data([3, 2], [4, 3], [5, 3])
    @unpack
    def test_list_extracted_into_arguments(self, first_value, second_value):
        self.assertTrue(first_value > second_value)

    @unpack
    @data({'first': 1, 'second': 3, 'third': 2},
          {'first': 4, 'second': 6, 'third': 5})
    def test_dicts_extracted_into_kwargs(self, first, second, third):
        self.assertTrue(first < third < second)
        
if __name__=='__main__':
    unittest.main(verbosity=2)
複製代碼

執行以後,所有pass。

3. 使用json文件

新建文件 test_data_list.json:

[
    "Hello",
    "Goodbye"
]

新建文件  test_data_dict.json:

{
    "unsorted_list": [ 10, 12, 15 ],
    "sorted_list": [ 15, 12, 50 ]
}

新建測試腳本ddt_test.py:

複製代碼
import unittest
from ddt import ddt, file_data
from ddt_demo.mycode import has_three_elements,is_a_greeting

@ddt
class FooTestCase(unittest.TestCase):

    @file_data('test_data_dict.json')
    def test_file_data_json_dict(self, value):
        self.assertTrue(has_three_elements(value))

    @file_data('test_data_list.json')
    def test_file_data_json_list(self, value):
        self.assertTrue(is_a_greeting(value))
        
if __name__=='__main__':
    unittest.main(verbosity=2)
複製代碼

4. 使用yaml文件

新建文件 test_data_list.yaml:

- "Hello"
- "Goodbye"

新建文件 test_data_dict.yaml:

複製代碼
unsorted_list:
  - 10
  - 15
  - 12

sorted_list: [ 15, 12, 50 ]
複製代碼

新建測試腳本ddt_test.py:

複製代碼
import unittest
from ddt import ddt, file_data
from ddt_demo.mycode import has_three_elements,is_a_greeting

@ddt
class FooTestCase(unittest.TestCase):

    @file_data('test_data_dict.yaml')
    def test_file_data_yaml_dict(self, value):
        self.assertTrue(has_three_elements(value))

    @file_data('test_data_list.yaml')
    def test_file_data_yaml_list(self, value):
        self.assertTrue(is_a_greeting(value))
        
if __name__=='__main__':
    unittest.main(verbosity=2)
複製代碼
相關文章
相關標籤/搜索