unittest --- 單元測試框架

原始碼:Lib/unittest/__init__.py


(假如你已經熟悉相關基礎的測試概念,你可能會希望跳過以下段落,直接參考 assert 方法清單。)

unittest 原生的單元測試框架最初由 JUnit 開發,和其他程式語言相似有主要的單元測試框架。支援自動化測試,對測試分享安裝與關閉程式碼,集合所有匯總的測試,並且獨立各個測試報告框架。

unittest 用來作為實現支援一些重要的物件導向方法的概念:

test fixture

一個 test fixture 代表執行一個或多個測試所需要的準備,以及其他相關清理操作,例如可以是建立臨時性的或是代理用 (proxy) 資料庫、目錄、或是啟動一個伺服器程序。

test case(測試用例)

一個 test case 是一個獨立的單元測試。這是用來確認一個特定設定的輸入的特殊回饋。 unittest 提供一個基礎類別,類別 TestCase,可以用來建立一個新的測試條例。

test suite(測試套件)

test suite 是一個搜集測試條例,測試套件,或是兩者皆有。它需要一起被執行並用來匯總測試。

test runner(測試執行器)

test runner 是一個編排測試執行與提供結果給使用者的一個元件。執行器可以使用圖形化介面,文字介面或是回傳一個特別值用來標示出執行測試的結果。

也參考

doctest 模組

另一個執行測試的模組,但使用不一樣的測試方法與規範。

Simple Smalltalk Testing: With Patterns

Kent Beck 的原始論文討論使用 unittest 這樣模式的測試框架。

pytest

第三方的單元測試框架,但在撰寫測試時使用更輕量的語法。例如: assert func(10) == 42

The Python Testing Tools Taxonomy

一份詳細的 Python 測試工具列表,包含 functional testing 框架和mock object 函式庫。

Testing in Python Mailing List

一個專門興趣的群組用來討論 Python 中的測試方式與測試工具。

随 Python 源代码分发的脚本 Tools/unittestgui/unittestgui.py 是一个一个用于发现和执行测试的用户图形界面工具。这主要是为了方便新手使用单元测试制作的。在生产环境中,测试应由一个持续集成(CI)系统运行,如 Buildbot, JenkinsTravis-CI, 或 AppVeyor

簡單範例

unittest 模組提供一系列豐富的工具用來建構與執行測試。本節將展示這一系列工具中一部份,它們已能滿足大部份使用者需求。

這是一段簡短的腳本用來測試 3 個字串方法:

import unittest

class TestStringMethods(unittest.TestCase):

    def test_upper(self):
        self.assertEqual('foo'.upper(), 'FOO')

    def test_isupper(self):
        self.assertTrue('FOO'.isupper())
        self.assertFalse('Foo'.isupper())

    def test_split(self):
        s = 'hello world'
        self.assertEqual(s.split(), ['hello', 'world'])
        # check that s.split fails when the separator is not a string
        with self.assertRaises(TypeError):
            s.split(2)

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

測試用例 (testcase) 可以透過繼承 unittest.TestCase 類別來建立。這裡定義了三個獨立的物件方法,名稱皆以 test 開頭。這樣的命名方式能告知 test runner 哪些物件方法為定義的測試。

每個測試的關鍵為呼叫 assertEqual() 來確認是否為期望的結果; assertTrue() 或是 assertFalse() 用來驗證一個條件式; assertRaises() 用來驗證是否觸發一個特定的 exception。使用這些物件方法來取代 assert 陳述句,將能使 test runner 收集所有的測試結果並產生一個報表。

通过 setUp()tearDown() 方法,可以设置测试开始前与完成后需要执行的指令。 在 组织你的测试代码 中,对此有更为详细的描述。

最後將顯示一個簡單的方法去執行測試 unittest.main() 提供一個命令執行列介面測試腳本。當透過命令執行列執行,輸出結果將會像是:

...
----------------------------------------------------------------------
Ran 3 tests in 0.000s

OK

在測試時加入 -v 選項將指示 unittest.main() 提高 verbosity 層級,產生以下的輸出:

test_isupper (__main__.TestStringMethods) ... ok
test_split (__main__.TestStringMethods) ... ok
test_upper (__main__.TestStringMethods) ... ok

----------------------------------------------------------------------
Ran 3 tests in 0.001s

OK

以上的例子顯示大多數使用 unittest 特徵足以滿足大多數日常測試的需求。接下來第一部分文件的剩餘部分將繼續探索完整特徵設定。

命令執行列介面 (Command-Line Interface)

單元測試模組可以透過命令執行列執行測試模組,物件甚至個別的測試方法:

python -m unittest test_module1 test_module2
python -m unittest test_module.TestClass
python -m unittest test_module.TestClass.test_method

你可以通過一個串列與任何模組名稱的組合,完全符合類別與方法的名稱。

測試模組可以根據檔案路徑指定:

python -m unittest tests/test_something.py

這允許你使用 shell 檔案名稱補完功能 (filename completion) 來指定測試模組。給定的檔案路徑必須亦能被當作模組 import。此路徑轉換為模組名稱的方式為移除 '.py' 並將路徑分隔符 (path separator) 轉換成 '.'。 假如你的測試檔案無法被 import 成模組,你應該直接執行該測試檔案。

通過增加 -v 的旗標數,可以在你執行測試時得到更多細節(更高的 verbosity):

python -m unittest -v test_module

若執行時不代任何引數,將執行 Test Discovery(測試探索)

python -m unittest

列出所有命令列選項:

python -m unittest -h

3.2 版更變: 在早期的版本可以個別執行測試方法和不需要模組或是類別。

命令列模式選項

unittest 支援以下命令列選項:

-b, --buffer

Standard output 與 standard error stream 將在測試執行被緩衝 (buffer)。這些輸出在測試通過時被丟棄。若是測試錯誤或失則,這些輸出將會正常地被印出,並且被加入至錯誤訊息中。

-c, --catch

Control-C 測試執行過程中等待正確的測試結果並回報目前為止所有的測試結果。第二個 Control-C 拋出一般例外 KeyboardInterrupt

參照 Signal Handling 針對函式提供的功能。

-f, --failfast

在第一次錯誤或是失敗停止執行測試。

-k

只运行匹配模式或子字符串的测试方法和类。 此选项可以被多次使用,在此情况下将会包括匹配任何给定模式的所有测试用例。

包含通配符(*)的模式使用 fnmatch.fnmatchcase() 对测试名称进行匹配。另外,该匹配是大小写敏感的。

模式对测试加载器导入的测试方法全名进行匹配。

例如,-k foo 可以匹配到 foo_tests.SomeTest.test_somethingbar_tests.SomeTest.test_foo ,但是不能匹配到 bar_tests.FooTest.test_something

--locals

透過 traceback 顯示本地變數。

3.2 版新加入: 增加命令列模式選項 -b-c-f

3.5 版新加入: 命令列選項 --locals

3.7 版新加入: 命令列選項 -k

對執行所有的專案或是一個子集合測試,命令列模式可以可以被用來做測試探索。

Test Discovery(測試探索)

3.2 版新加入.

單元測試支援簡單的 test discovery(測試探索)。為了相容於測試探索,所有的測試檔案都要是模組或是套件(包含 namespace packages),並能從專案的最上層目錄中 import(代表它們的檔案名稱必須是有效的 identifiers)。

Test discovery(測試探索)實作在 TestLoader.discover(),但也可以被用於命令列模式。基本的命令列模式用法如下:

cd project_directory
python -m unittest discover

備註

python -m unittest 作為捷徑,其功能相當於 python -m unittest discover。假如你想傳遞引數至探索測試的話,一定要明確地加入 discover 子指令。

discover 子指令有以下幾個選項:

-v, --verbose

詳細(verbose)輸出

-s, --start-directory directory

開始尋找的資料夾(預設為 .

-p, --pattern pattern

匹配測試檔案的模式(預設為 test*.py

-t, --top-level-directory directory

專案的最高階層目錄 (defaults to start directory)

-s, -p, 和 -t 選項依照傳遞位置作為引數排序順序。以下兩個命令列被視為等價:

python -m unittest discover -s project_directory -p "*_test.py"
python -m unittest discover project_directory "*_test.py"

正如可以传入路径那样,传入一个包名作为起始目录也是可行的,如 myproject.subpackage.test 。你提供的包名会被导入,它在文件系统中的位置会被作为起始目录。

警示

探索性测试通过导入测试对测试进行加载。在找到所有你指定的开始目录下的所有测试文件后,它把路径转换为包名并进行导入。如 foo/bar/baz.py 会被导入为 foo.bar.baz

如果你有一个全局安装的包,并尝试对这个包的副本进行探索性测试,可能会从错误的地方开始导入。如果出现这种情况,测试会输出警告并退出。

如果你使用包名而不是路径作为开始目录,搜索时会假定它导入的是你想要的目录,所以你不会收到警告。

测试模块和包可以通过 load_tests protocol 自定义测试的加载和搜索。

3.4 版更變: 测试发现支持初始目录下的 命名空间包。注意你也需要指定顶层目录(例如:python -m unittest discover -s root/namespace -t root)。

组织你的测试代码

单元测试的构建单位是 test cases :独立的、包含执行条件与正确性检查的方案。在 unittest 中,测试用例表示为 unittest.TestCase 的实例。通过编写 TestCase 的子类或使用 FunctionTestCase 编写你自己的测试用例。

一个 TestCase 实例的测试代码必须是完全自含的,因此它可以独立运行,或与其它任意组合任意数量的测试用例一起运行。

TestCase 的最简单的子类需要实现一个测试方法(例如一个命名以 test 开头的方法)以执行特定的测试代码:

import unittest

class DefaultWidgetSizeTestCase(unittest.TestCase):
    def test_default_widget_size(self):
        widget = Widget('The widget')
        self.assertEqual(widget.size(), (50, 50))

可以看到,为了进行测试,我们使用了基类 TestCase 提供的其中一个 assert*() 方法。若测试不通过,将会引发一个带有说明信息的异常,并且 unittest 会将这个测试用例标记为测试不通过。任何其它类型的异常将会被当做错误处理。

可能同时存在多个前置操作相同的测试,我们可以把测试的前置操作从测试代码中拆解出来,并实现测试前置方法 setUp() 。在运行测试时,测试框架会自动地为每个单独测试调用前置方法。

import unittest

class WidgetTestCase(unittest.TestCase):
    def setUp(self):
        self.widget = Widget('The widget')

    def test_default_widget_size(self):
        self.assertEqual(self.widget.size(), (50,50),
                         'incorrect default size')

    def test_widget_resize(self):
        self.widget.resize(100,150)
        self.assertEqual(self.widget.size(), (100,150),
                         'wrong size after resize')

備註

多个测试运行的顺序由内置字符串排序方法对测试名进行排序的结果决定。

在测试运行时,若 setUp() 方法引发异常,测试框架会认为测试发生了错误,因此测试方法不会被运行。

相似的,我们提供了一个 tearDown() 方法在测试方法运行后进行清理工作。

import unittest

class WidgetTestCase(unittest.TestCase):
    def setUp(self):
        self.widget = Widget('The widget')

    def tearDown(self):
        self.widget.dispose()

setUp() 成功运行,无论测试方法是否成功,都会运行 tearDown()

这样的一个测试代码运行的环境被称为 test fixture 。一个新的 TestCase 实例作为一个测试脚手架,用于运行各个独立的测试方法。在运行每个测试时,setUp()tearDown()__init__() 会被调用一次。

推荐你根据用例所测试的功能将测试用 TestCase 分组。unittest 为此提供了 test suiteunittestTestSuite 类是一个代表。通常情况下,调用 unittest.main() 就能正确地找到并执行这个模块下所有用 TestCase 分组的测试。

然而,如果你需要自定义你的测试套件的话,你可以参考以下方法组织你的测试:

def suite():
    suite = unittest.TestSuite()
    suite.addTest(WidgetTestCase('test_default_widget_size'))
    suite.addTest(WidgetTestCase('test_widget_resize'))
    return suite

if __name__ == '__main__':
    runner = unittest.TextTestRunner()
    runner.run(suite())

你可以把测试用例和测试套件放在与被测试代码相同的模块中(比如 widget.py),但将测试代码放在单独的模块中(比如 test_widget.py)有几个优势。

  • 测试模块可以从命令行被独立调用。

  • 更容易在分发的代码中剥离测试代码。

  • 降低没有好理由的情况下修改测试代码以通过测试的诱惑。

  • 测试代码应比被测试代码更少地被修改。

  • 被测试代码可以更容易地被重构。

  • 对用 C 语言写成的模块无论如何都得单独写成一个模块,为什么不保持一致呢?

  • 如果测试策略发生了改变,没有必要修改源代码。

复用已有的测试代码

一些用户希望直接使用 unittest 运行已有的测试代码,而不需要把已有的每个测试函数转化为一个 TestCase 的子类。

因此, unittest 提供 FunctionTestCase 类。这个 TestCase 的子类可用于打包已有的测试函数,并支持设置前置与后置函数。

假定有一个测试函数:

def testSomething():
    something = makeSomething()
    assert something.name is not None
    # ...

可以创建等价的测试用例如下,其中前置和后置方法是可选的。

testcase = unittest.FunctionTestCase(testSomething,
                                     setUp=makeSomethingDB,
                                     tearDown=deleteSomethingDB)

備註

FunctionTestCase 可以快速将现有的测试转换成基于 unittest 的测试,但不推荐你这样做。花点时间继承 TestCase 会让以后重构测试无比轻松。

在某些情况下,现有的测试可能是用 doctest 模块编写的。 如果是这样, doctest 提供了一个 DocTestSuite 类,可以从现有基于 doctest 的测试中自动构建 unittest.TestSuite 用例。

跳过测试与预计的失败

3.1 版新加入.

Unittest 支持跳过单个或整组的测试用例。它还支持把测试标注成“预期失败”的测试。这些坏测试会失败,但不会算进 TestResult 的失败里。

要跳过测试只需使用 skip() decorator 或其附带条件的版本,在 setUp() 内部使用 TestCase.skipTest(),或是直接引发 SkipTest

跳过测试的基本用法如下:

class MyTestCase(unittest.TestCase):

    @unittest.skip("demonstrating skipping")
    def test_nothing(self):
        self.fail("shouldn't happen")

    @unittest.skipIf(mylib.__version__ < (1, 3),
                     "not supported in this library version")
    def test_format(self):
        # Tests that work for only a certain version of the library.
        pass

    @unittest.skipUnless(sys.platform.startswith("win"), "requires Windows")
    def test_windows_support(self):
        # windows specific testing code
        pass

    def test_maybe_skipped(self):
        if not external_resource_available():
            self.skipTest("external resource not available")
        # test code that depends on the external resource
        pass

在啰嗦模式下运行以上测试例子时,程序输出如下:

test_format (__main__.MyTestCase) ... skipped 'not supported in this library version'
test_nothing (__main__.MyTestCase) ... skipped 'demonstrating skipping'
test_maybe_skipped (__main__.MyTestCase) ... skipped 'external resource not available'
test_windows_support (__main__.MyTestCase) ... skipped 'requires Windows'

----------------------------------------------------------------------
Ran 4 tests in 0.005s

OK (skipped=4)

跳过测试类的写法跟跳过测试方法的写法相似:

@unittest.skip("showing class skipping")
class MySkippedTestCase(unittest.TestCase):
    def test_not_run(self):
        pass

TestCase.setUp() 也可以跳过测试。可以用于所需资源不可用的情况下跳过接下来的测试。

使用 expectedFailure() 装饰器表明这个测试预计失败。:

class ExpectedFailureTestCase(unittest.TestCase):
    @unittest.expectedFailure
    def test_fail(self):
        self.assertEqual(1, 0, "broken")

你可以很容易地编写在测试时调用 skip() 的装饰器作为自定义的跳过测试装饰器。 下面这个装饰器会跳过测试,除非所传入的对象具有特定的属性:

def skipUnlessHasattr(obj, attr):
    if hasattr(obj, attr):
        return lambda func: func
    return unittest.skip("{!r} doesn't have {!r}".format(obj, attr))

以下的装饰器和异常实现了跳过测试和预期失败两种功能:

@unittest.skip(reason)

跳过被此装饰器装饰的测试。 reason 为测试被跳过的原因。

@unittest.skipIf(condition, reason)

condition 为真时,跳过被装饰的测试。

@unittest.skipUnless(condition, reason)

跳过被装饰的测试,除非 condition 为真。

@unittest.expectedFailure

将测试标记为预期的失败或错误。 如果测试失败或在测试函数自身(而非在某个 test fixture 方法)中出现错误则将认为是测试成功。 如果测试通过,则将认为是测试失败。

exception unittest.SkipTest(reason)

引发此异常以跳过一个测试。

通常来说,你可以使用 TestCase.skipTest() 或其中一个跳过测试的装饰器实现跳过测试的功能,而不是直接引发此异常。

被跳过的测试的 setUp()tearDown() 不会被运行。被跳过的类的 setUpClass()tearDownClass() 不会被运行。被跳过的模组的 setUpModule()tearDownModule() 不会被运行。

使用子测试区分测试迭代

3.4 版新加入.

当你的几个测试之间的差异非常小,例如只有某些形参不同时,unittest 允许你使用 subTest() 上下文管理器在一个测试方法体的内部区分它们。

舉例來說,以下測試:

class NumbersTest(unittest.TestCase):

    def test_even(self):
        """
        Test that numbers between 0 and 5 are all even.
        """
        for i in range(0, 6):
            with self.subTest(i=i):
                self.assertEqual(i % 2, 0)

會有以下輸出:

======================================================================
FAIL: test_even (__main__.NumbersTest) (i=1)
----------------------------------------------------------------------
Traceback (most recent call last):
  File "subtests.py", line 32, in test_even
    self.assertEqual(i % 2, 0)
AssertionError: 1 != 0

======================================================================
FAIL: test_even (__main__.NumbersTest) (i=3)
----------------------------------------------------------------------
Traceback (most recent call last):
  File "subtests.py", line 32, in test_even
    self.assertEqual(i % 2, 0)
AssertionError: 1 != 0

======================================================================
FAIL: test_even (__main__.NumbersTest) (i=5)
----------------------------------------------------------------------
Traceback (most recent call last):
  File "subtests.py", line 32, in test_even
    self.assertEqual(i % 2, 0)
AssertionError: 1 != 0

如果不使用子测试,程序遇到第一次错误之后就会停止。而且因为 i 的值不显示,错误也更难找。

======================================================================
FAIL: test_even (__main__.NumbersTest)
----------------------------------------------------------------------
Traceback (most recent call last):
  File "subtests.py", line 32, in test_even
    self.assertEqual(i % 2, 0)
AssertionError: 1 != 0

类与函数

本节深入介绍了 unittest 的 API。

测试用例

class unittest.TestCase(methodName='runTest')

TestCase 类的实例代表了 unittest 宇宙中的逻辑测试单元。 该类旨在被当作基类使用,特定的测试将由其实体子类来实现。 该类实现了测试运行器所需的接口以允许它驱动测试,并实现了可被测试代码用来检测和报告各种类型的失败的方法。

每个 TestCase 实例将运行一个单位的基础方法:即名为 methodName 的方法。 在使用 TestCase 的大多数场景中,你都不需要修改 methodName 或重新实现默认的 runTest() 方法。

3.2 版更變: TestCase 不需要提供 methodName 即可成功实例化。 这使得从交互式解释器试验 TestCase 更为容易。

TestCase 的实例提供了三组方法:一组用来运行测试,另一组被测试实现用来检查条件和报告失败,还有一些查询方法用来收集有关测试本身的信息。

第一组(用于运行测试的)方法是:

setUp()

为测试预备而调用的方法。 此方法会在调用测试方法之前被调用;除了 AssertionErrorSkipTest,此方法所引发的任何异常都将被视为错误而非测试失败。 默认的实现将不做任何事情。

tearDown()

在测试方法被调用并记录结果之后立即被调用的方法。 此方法即使在测试方法引发异常时仍会被调用,因此子类中的实现将需要特别注意检查内部状态。 除 AssertionErrorSkipTest 外,此方法所引发的任何异常都将被视为额外的错误而非测试失败(因而会增加总计错误报告数)。 此方法将只在 setUp() 成功执行时被调用,无论测试方法的结果如何。 默认的实现将不做任何事情。

setUpClass()

在一个单独类中的测试运行之前被调用的类方法。 setUpClass 会被作为唯一的参数在类上调用且必须使用 classmethod() 装饰器:

@classmethod
def setUpClass(cls):
    ...

更多細節請見 Class and Module Fixtures

3.2 版新加入.

tearDownClass()

在一个单独类的测试完成运行之后被调用的类方法。 tearDownClass 会被作为唯一的参数在类上调用且必须使用 classmethod() 装饰器:

@classmethod
def tearDownClass(cls):
    ...

更多細節請見 Class and Module Fixtures

3.2 版新加入.

run(result=None)

运行测试,将结果收集至作为 result 传入的 TestResult。 如果 result 被省略或为 None,则会创建一个临时的结果对象(通过调用 defaultTestResult() 方法)并使用它。 结果对象会被返回给 run() 的调用方。

同样的效果也可通过简单地调用 TestCase 实例来达成。

3.3 版更變: 之前版本的 run 不会返回结果。 也不会对实例执行调用。

skipTest(reason)

在测试方法或 setUp() 执行期间调用此方法将跳过当前测试。 详情参见 跳过测试与预计的失败

3.1 版新加入.

subTest(msg=None, **params)

返回一个上下文管理器以将其中的代码块作为子测试来执行。 可选的 msgparams 是将在子测试失败时显示的任意值,以便让你能清楚地标识它们。

一个测试用例可以包含任意数量的子测试声明,并且它们可以任意地嵌套。

更多資訊請見 使用子测试区分测试迭代

3.4 版新加入.

debug()

运行测试而不收集结果。 这允许测试所引发的异常被传递给调用方,并可被用于支持在调试器中运行测试。

TestCase 类提供了一些断言方法用于检查并报告失败。 下表列出了最常用的方法(请查看下文的其他表来了解更多的断言方法):

方法

检查对象

引入版本

assertEqual(a, b)

a == b

assertNotEqual(a, b)

a != b

assertTrue(x)

bool(x) is True

assertFalse(x)

bool(x) is False

assertIs(a, b)

a is b

3.1

assertIsNot(a, b)

a is not b

3.1

assertIsNone(x)

x is None

3.1

assertIsNotNone(x)

x is not None

3.1

assertIn(a, b)

a in b

3.1

assertNotIn(a, b)

a not in b

3.1

assertIsInstance(a, b)

isinstance(a, b)

3.2

assertNotIsInstance(a, b)

not isinstance(a, b)

3.2

这些断言方法都支持 msg 参数,如果指定了该参数,它将被用作测试失败时的错误消息 (另请参阅 longMessage)。 请注意将 msg 关键字参数传给 assertRaises(), assertRaisesRegex(), assertWarns(), assertWarnsRegex() 的前提是它们必须被用作上下文管理器。

assertEqual(first, second, msg=None)

测试 firstsecond 是否相等。 如果两个值的比较结果是不相等,则测试将失败。

此外,如果 firstsecond 的类型完全相同且属于 list, tuple, dict, set, frozenset 或 str 或者属于通过 addTypeEqualityFunc() 注册子类的类型则将会调用类型专属的相等判断函数以便生成更有用的默认错误消息 (另请参阅 类型专属方法列表)。

3.1 版更變: 增加了对类型专属的相等判断函数的自动调用。

3.2 版更變: 增加了 assertMultiLineEqual() 作为用于比较字符串的默认类型相等判断函数。

assertNotEqual(first, second, msg=None)

测试 firstsecond 是否不等。 如果两个值的比较结果是相等,则测试将失败。

assertTrue(expr, msg=None)
assertFalse(expr, msg=None)

测试 expr 是否为真值(或假值)。

请注意这等价于 bool(expr) is True 而不等价于 expr is True (后者要使用 assertIs(expr, True))。 当存在更专门的方法时也应避免使用此方法 (例如应使用 assertEqual(a, b) 而不是 assertTrue(a == b)),因为它们在测试失败时会提供更有用的错误消息。

assertIs(first, second, msg=None)
assertIsNot(first, second, msg=None)

测试 firstsecond 是 (或不是) 同一个对象。

3.1 版新加入.

assertIsNone(expr, msg=None)
assertIsNotNone(expr, msg=None)

测试 expr 是 (或不是) None

3.1 版新加入.

assertIn(member, container, msg=None)
assertNotIn(member, container, msg=None)

测试 member 是 (或不是) container 的成员。

3.1 版新加入.

assertIsInstance(obj, cls, msg=None)
assertNotIsInstance(obj, cls, msg=None)

测试 obj 是 (或不是) cls (此参数可以为一个类或包含类的元组,即 isinstance() 所接受的参数) 的实例。 要检测是否为指定类型,请使用 assertIs(type(obj), cls)

3.2 版新加入.

还可以使用下列方法来检查异常、警告和日志消息的产生:

方法

检查对象

引入版本

assertRaises(exc, fun, *args, **kwds)

fun(*args, **kwds) 引发了 exc

assertRaisesRegex(exc, r, fun, *args, **kwds)

fun(*args, **kwds) 引发了 exc 并且消息可与正则表达式 r 相匹配

3.1

assertWarns(warn, fun, *args, **kwds)

fun(*args, **kwds) 引发了 warn

3.2

assertWarnsRegex(warn, r, fun, *args, **kwds)

fun(*args, **kwds) 引发了 warn 并且消息可与正则表达式 r 相匹配

3.2

assertLogs(logger, level)

with 代码块在 logger 上使用了最小的 level 级别写入日志

3.4

assertNoLogs(logger, level)

with 代码块没有在

logger 上使用最小的 level 级别写入日志

3.10

assertRaises(exception, callable, *args, **kwds)
assertRaises(exception, *, msg=None)

测试当 callable 附带任何同时被传给 assertRaises() 的位置或关键字参数被调用时是否引发了异常。 如果引发了 exception 则测试通过,如果引发了另一个异常则报错,或者如果未引发任何异常则测试失败。 要捕获一组异常中的任何一个,可以将包含多个异常类的元组作为 exception 传入。

如果只给出了 exception 和可能的 msg 参数,则返回一个上下文管理器以便被测试的代码可以被写成内联形式而不是被写成函数:

with self.assertRaises(SomeException):
    do_something()

当被作为上下文管理器使用时,assertRaises() 接受额外的关键字参数 msg

上下文管理器将把捕获的异常对象存入在其 exception 属性中。 这适用于需要对所引发异常执行额外检查的场合:

with self.assertRaises(SomeException) as cm:
    do_something()

the_exception = cm.exception
self.assertEqual(the_exception.error_code, 3)

3.1 版更變: 添加了将 assertRaises() 用作上下文管理器的功能。

3.2 版更變: 新增 exception 屬性。

3.3 版更變: 增加了 msg 关键字参数在作为上下文管理器时使用。

assertRaisesRegex(exception, regex, callable, *args, **kwds)
assertRaisesRegex(exception, regex, *, msg=None)

assertRaises() 类似但还会测试 regex 是否匹配被引发异常的字符串表示形式。 regex 可以是一个正则表达式对象或包含正则表达式的字符串以提供给 re.search() 使用。 例如:

self.assertRaisesRegex(ValueError, "invalid literal for.*XYZ'$",
                       int, 'XYZ')

或是:

with self.assertRaisesRegex(ValueError, 'literal'):
   int('XYZ')

3.1 版新加入: 以方法名 assertRaisesRegexp 添加。

3.2 版更變: 重新命名為 assertRaisesRegex()

3.3 版更變: 增加了 msg 关键字参数在作为上下文管理器时使用。

assertWarns(warning, callable, *args, **kwds)
assertWarns(warning, *, msg=None)

测试当 callable 附带任何同时被传给 assertWarns() 的位置或关键字参数被调用时是否触发了警告。 如果触发了 warning 则测试通过,否则测试失败。 引发任何异常则报错。 要捕获一组警告中的任何一个,可将包含多个警告类的元组作为 warnings 传入。

如果只给出了 warning 和可能的 msg 参数,则返回一个上下文管理器以便被测试的代码可以被写成内联形式而不是被写成函数:

with self.assertWarns(SomeWarning):
    do_something()

当被作为上下文管理器使用时,assertWarns() 接受额外的关键字参数 msg

上下文管理器将把捕获的警告对象保存在其 warning 属性中,并把触发警告的源代码行保存在 filenamelineno 属性中。 这适用于需要对捕获的警告执行额外检查的场合:

with self.assertWarns(SomeWarning) as cm:
    do_something()

self.assertIn('myfile.py', cm.filename)
self.assertEqual(320, cm.lineno)

无论被调用时警告过滤器是否就位此方法均可工作。

3.2 版新加入.

3.3 版更變: 增加了 msg 关键字参数在作为上下文管理器时使用。

assertWarnsRegex(warning, regex, callable, *args, **kwds)
assertWarnsRegex(warning, regex, *, msg=None)

assertWarns() 类似但还会测试 regex 是否匹配被触发警告的消息文本。 regex 可以是一个正则表达式对象或包含正则表达式的字符串以提供给 re.search() 使用。 例如:

self.assertWarnsRegex(DeprecationWarning,
                      r'legacy_function\(\) is deprecated',
                      legacy_function, 'XYZ')

或是:

with self.assertWarnsRegex(RuntimeWarning, 'unsafe frobnicating'):
    frobnicate('/etc/passwd')

3.2 版新加入.

3.3 版更變: 增加了 msg 关键字参数在作为上下文管理器时使用。

assertLogs(logger=None, level=None)

一个上下文管理器,它测试在 logger 或其子对象上是否至少记录了一条至少为指定 level 以上级别的消息。

如果给出了 logger 则它应为一个 logging.Logger 对象或为一个指定日志记录器名称的 str。 默认为根日志记录器,它将捕获未被非传播型后继日志记录器所拦阻的所有消息。

如果给出了 level 则它应为一个用数字表示的日志记录级别或其字符串形式 (例如 "ERROR"logging.ERROR)。 默认为 logging.INFO

如果在 with 代码块内部发出了至少一条与 loggerlevel 条件相匹配的消息则测试通过,否则测试失败。

上下文管理器返回的对象是一个记录辅助器,它会记录所匹配的日志消息。 它有两个属性:

records

所匹配的日志消息 logging.LogRecord 对象组成的列表。

output

str 对象组成的列表,内容为所匹配消息经格式化后的输出。

範例:

with self.assertLogs('foo', level='INFO') as cm:
    logging.getLogger('foo').info('first message')
    logging.getLogger('foo.bar').error('second message')
self.assertEqual(cm.output, ['INFO:foo:first message',
                             'ERROR:foo.bar:second message'])

3.4 版新加入.

assertNoLogs(logger=None, level=None)

一个上下文管理器,它测试在 logger 或其子对象上是否未记录任何至少为指定 level 以上级别的消息。

如果给出了 logger 则它应为一个 logging.Logger 对象或为一个指定日志记录器名称的 str。 默认为根日志记录器,它将捕获所有消息。

如果给出了 level 则它应为一个用数字表示的日志记录级别或其字符串形式 (例如 "ERROR"logging.ERROR)。 默认为 logging.INFO

assertLogs() 不同,上下文管理器将不返回任何对象。

3.10 版新加入.

还有其他一些方法可用于执行更专门的检查,例如:

方法

检查对象

引入版本

assertAlmostEqual(a, b)

round(a-b, 7) == 0

assertNotAlmostEqual(a, b)

round(a-b, 7) != 0

assertGreater(a, b)

a > b

3.1

assertGreaterEqual(a, b)

a >= b

3.1

assertLess(a, b)

a < b

3.1

assertLessEqual(a, b)

a <= b

3.1

assertRegex(s, r)

r.search(s)

3.1

assertNotRegex(s, r)

not r.search(s)

3.2

assertCountEqual(a, b)

ab 具有同样数量的相同元素,无论其顺序如何。

3.2

assertAlmostEqual(first, second, places=7, msg=None, delta=None)
assertNotAlmostEqual(first, second, places=7, msg=None, delta=None)

测试 firstsecond 是否几乎相等,比较的标准是计算差值并舍入到 places 所指定的十进制位数 (默认为 7 位),再与零相比较。 请注意此方法是将结果值舍入到指定的 十进制位数 (即相当于 round() 函数) 而非 有效位数

如果提供了 delta 而非 placesfirstsecond 之间的差值必须小于等于 (或大于) delta

同时提供 deltaplaces 将引发 TypeError

3.2 版更變: assertAlmostEqual() 会自动将几乎相等的对象视为相等。 而如果对象相等则 assertNotAlmostEqual() 会自动测试失败。 增加了 delta 关键字参数。

assertGreater(first, second, msg=None)
assertGreaterEqual(first, second, msg=None)
assertLess(first, second, msg=None)
assertLessEqual(first, second, msg=None)

根据方法名分别测试 first 是否 >, >=, < 或 <= second。 如果不是,则测试失败:

>>> self.assertGreaterEqual(3, 4)
AssertionError: "3" unexpectedly not greater than or equal to "4"

3.1 版新加入.

assertRegex(text, regex, msg=None)
assertNotRegex(text, regex, msg=None)

测试一个 regex 搜索匹配(或不匹配) 文本。如果不匹配,错误信息中将包含匹配模式和 文本*(或部分匹配失败的 *文本)。regex 可以是正则表达式对象或能够用于 re.search() 的包含正则表达式的字符串。

3.1 版新加入: 以方法名 assertRegexpMatches 添加。

3.2 版更變: 方法 assertRegexpMatches() 已被改名为 assertRegex()

3.2 版新加入: assertNotRegex()

3.5 版新加入: assertNotRegexpMatches 这个名字是 assertNotRegex() 的已被弃用的别名。

assertCountEqual(first, second, msg=None)

测试序列 firstsecond 是否包含同样的元素,无论其顺序如何。 当存在差异时,将生成一条错误消息来列出两个序列之间的差异。

重复的元素 不会firstsecond 的比较中被忽略。 它会检查每个元素在两个序列中的出现次数是否相同。 等价于: assertEqual(Counter(list(first)), Counter(list(second))) 但还适用于包含不可哈希对象的序列。

3.2 版新加入.

assertEqual() 方法会将相同类型对象的相等性检查分派给不同的类型专属方法。 这些方法已被大多数内置类型所实现,但也可以使用 addTypeEqualityFunc() 来注册新的方法:

addTypeEqualityFunc(typeobj, function)

注册一个由 assertEqual() 调用的特定类型专属方法来检查恰好为相同 typeobj (而非子类) 的两个对象是否相等。 function 必须接受两个位置参数和第三个 msg=None 关键字参数,就像 assertEqual() 那样。 当检测到前两个形参之间不相等时它必须引发 self.failureException(msg) -- 可能还会提供有用的信息并在错误消息中详细解释不相等的原因。

3.1 版新加入.

以下是 assertEqual() 自动选用的不同类型的比较方法。一般情况下不需要直接在测试中调用这些方法。

方法

用作比较

引入版本

assertMultiLineEqual(a, b)

字符串

3.1

assertSequenceEqual(a, b)

序列

3.1

assertListEqual(a, b)

列表

3.1

assertTupleEqual(a, b)

元组

3.1

assertSetEqual(a, b)

集合

3.1

assertDictEqual(a, b)

字典

3.1

assertMultiLineEqual(first, second, msg=None)

测试多行字符串 first 是否与字符串 second 相等。 当不相等时将在错误消息中包括两个字符串之间差异的高亮显示。 此方法会在通过 assertEqual() 进行字符串比较时默认被使用。

3.1 版新加入.

assertSequenceEqual(first, second, msg=None, seq_type=None)

测试两个序列是否相等。 如果提供了 seq_type,则 firstsecond 都必须为 seq_type 的实例否则将引发失败。 如果两个序列不相等则会构造一个错误消息来显示两者之间的差异。

此方法不会被 assertEqual() 直接调用,但它会被用于实现 assertListEqual()assertTupleEqual()

3.1 版新加入.

assertListEqual(first, second, msg=None)
assertTupleEqual(first, second, msg=None)

测试两个列表或元组是否相等。 如果不相等,则会构造一个错误消息来显示两者之间的差异。 如果某个形参的类型不正确也会引发错误。 这些方法会在通过 assertEqual() 进行列表或元组比较时默认被使用。

3.1 版新加入.

assertSetEqual(first, second, msg=None)

测试两个集合是否相等。 如果不相等,则会构造一个错误消息来列出两者之间的差异。 此方法会在通过 assertEqual() 进行集合或冻结集合比较时默认被使用。

如果 firstsecond 没有 set.difference() 方法则测试失败。

3.1 版新加入.

assertDictEqual(first, second, msg=None)

测试两个字典是否相等。 如果不相等,则会构造一个错误消息来显示两个字典的差异。 此方法会在对 assertEqual() 的调用中默认被用来进行字典的比较。

3.1 版新加入.

最后 TestCase 还提供了以下的方法和属性:

fail(msg=None)

无条件地发出测试失败消息,附带错误消息 msgNone

failureException

这个类属性给出测试方法所引发的异常。 如果某个测试框架需要使用专门的异常,并可能附带额外的信息,则必须子类化该类以便与框架“正常互动”。 这个属性的初始值为 AssertionError

longMessage

这个类属性决定当将一个自定义失败消息作为 msg 参数传给一个失败的 assertXYY 调用时会发生什么。默认值为 True。 在此情况下,自定义消息会被添加到标准失败消息的末尾。 当设为 False 时,自定义消息会替换标准消息。

类设置可以通过在调用断言方法之前将一个实例属性 self.longMessage 赋值为 TrueFalse 在单个测试方法中进行重载。

类设置会在每个测试调用之前被重置。

3.1 版新加入.

maxDiff

这个属性控制来自在测试失败时报告 diffs 的断言方法的 diffs 输出的最大长度。 它默认为 80*8 个字符。 这个属性所影响的断言方法有 assertSequenceEqual() (包括所有委托给它的序列比较方法), assertDictEqual() 以及 assertMultiLineEqual()

maxDiff 设为 None 表示不限制 diffs 的最大长度。

3.2 版新加入.

测试框架可使用下列方法来收集测试的有关信息:

countTestCases()

返回此测试对象所提供的测试数量。 对于 TestCase 实例,该数量将总是为 1

defaultTestResult()

返回此测试类所要使用的测试结果类的实例(如果未向 run() 方法提供其他结果实例)。

对于 TestCase 实例,该返回值将总是为 TestResult 的实例;TestCase 的子类应当在有必要时重写此方法。

id()

返回一个标识指定测试用例的字符串。 该返回值通常为测试方法的完整名称,包括模块名和类名。

shortDescription()

返回测试的描述,如果未提供描述则返回 None。 此方法的默认实现将在可用的情况下返回测试方法的文档字符串的第一行,或者返回 None

3.1 版更變: 在 3.1 中已修改此方法将测试名称添加到简短描述中,即使存在文档字符串。 这导致了与单元测试扩展的兼容性问题因而在 Python 3.2 中将添加测试名称操作改到 TextTestResult 中。

addCleanup(function, /, *args, **kwargs)

tearDown() 之后添加了一个要调用的函数来清理测试期间所使用的资源。 函数将按它们被添加的相反顺序被调用 (LIFO)。 它们在调用时将附带它们被添加时传给 addCleanup() 的任何参数和关键字参数。

如果 setUp() 失败,即意味着 tearDown() 未被调用,则已添加的任何清理函数仍将被调用。

3.1 版新加入.

doCleanups()

此方法会在 tearDown() 之后,或者如果 setUp() 引发了异常则会在 setUp() 之后被调用。

它将负责调用由 addCleanup() 添加的所有清理函数。 如果你需要在 tearDown() 之前 调用清理函数则可以自行调用 doCleanups()

doCleanups() 每次会弹出清理函数栈中的一个方法,因此它可以在任何时候被调用。

3.1 版新加入.

classmethod addClassCleanup(function, /, *args, **kwargs)

在Add a function to be called after tearDownClass() 之后添加了一个要调用的函数来清理测试类运行期间所使用的资源。 函数将按它们被添加的相反顺序被调用 (LIFO)。 它们在调用时将附带它们被添加时传给 addClassCleanup() 的任何参数和关键字参数。

如果 setUpClass() 失败,即意味着 tearDownClass() 未被调用,则已添加的任何清理函数仍将被调用。

3.8 版新加入.

classmethod doClassCleanups()

此方法会在 tearDownClass() 之后无条件地被调用,或者如果 setUpClass() 引发了异常则会在 setUpClass() 之后被调用。

它将负责访问由 addClassCleanup() 添加的所有清理函数。 如果你需要在 tearDownClass() 之前 调用清理函数则可以自行调用 doClassCleanups()

doClassCleanups() 每次会弹出清理函数栈中的一个方法,因此它在任何时候被调用。

3.8 版新加入.

class unittest.IsolatedAsyncioTestCase(methodName='runTest')

这个类提供了与 TestCase 类似的 API 并也接受协程作为测试函数。

3.8 版新加入.

coroutine asyncSetUp()

为测试预备而调用的方法。 此方法会在 setUp() 之后被调用。 此方法将在调用测试方法之前立即被调用;除了 AssertionErrorSkipTest,此方法所引发的任何异常都将被视为错误而非测试失败。 默认的实现将不做任何事情。

coroutine asyncTearDown()

在测试方法被调用并记录结果之后立即被调用的方法。 此方法会在 tearDown() 之前被调用。 此方法即使在测试方法引发异常时仍会被调用,因此子类中的实现将需要特别注意检查内部状态。 除 AssertionErrorSkipTest 外,此方法所引发的任何异常都将被视为额外的错误而非测试失败(因而会增加总计错误报告数)。 此方法将只在 asyncSetUp() 成功执行时被调用,无论测试方法的结果如何。 默认的实现将不做任何事情。

addAsyncCleanup(function, /, *args, **kwargs)

此方法接受一个可被用作清理函数的协程。

run(result=None)

设置一个新的事件循环来运行测试,将结果收集至作为 result 传入的 TestResult。 如果 result 被省略或为 None,则会创建一个临时的结果对象(通过调用 defaultTestResult() 方法)并使用它。 结果对象会被返回给 run() 的调用方。 在测试结束时事件循环中的所有任务都将被取消。

一个显示先后顺序的例子:

from unittest import IsolatedAsyncioTestCase

events = []


class Test(IsolatedAsyncioTestCase):


    def setUp(self):
        events.append("setUp")

    async def asyncSetUp(self):
        self._async_connection = await AsyncConnection()
        events.append("asyncSetUp")

    async def test_response(self):
        events.append("test_response")
        response = await self._async_connection.get("https://example.com")
        self.assertEqual(response.status_code, 200)
        self.addAsyncCleanup(self.on_cleanup)

    def tearDown(self):
        events.append("tearDown")

    async def asyncTearDown(self):
        await self._async_connection.close()
        events.append("asyncTearDown")

    async def on_cleanup(self):
        events.append("cleanup")

if __name__ == "__main__":
    unittest.main()

在运行测试之后,events 将会包含 ["setUp", "asyncSetUp", "test_response", "asyncTearDown", "tearDown", "cleanup"]

class unittest.FunctionTestCase(testFunc, setUp=None, tearDown=None, description=None)

这个类实现了 TestCase 的部分接口,允许测试运行方驱动测试,但不提供可被测试代码用来检查和报告错误的方法。 这个类被用于创建使用传统测试代码的测试用例,允许它被集成到基于 unittest 的测试框架中。

一些已被弃用的别名

由于历史原因,某些 TestCase 方法具有一个或几个已目前已弃用的别名。 下表列出了它们的正确名称和已弃用的别名:

方法名

已弃用的别名

已弃用的别名

assertEqual()

failUnlessEqual

assertEquals

assertNotEqual()

failIfEqual

assertNotEquals

assertTrue()

failUnless

assert_

assertFalse()

failIf

assertRaises()

failUnlessRaises

assertAlmostEqual()

failUnlessAlmostEqual

assertAlmostEquals

assertNotAlmostEqual()

failIfAlmostEqual

assertNotAlmostEquals

assertRegex()

assertRegexpMatches

assertNotRegex()

assertNotRegexpMatches

assertRaisesRegex()

assertRaisesRegexp

3.1 版後已棄用: 在第二列中列出的 fail* 别名已经被弃用。

3.2 版後已棄用: 在第三列中列出的 assert* 别名已经被弃用。

3.2 版後已棄用: assertRegexpMatchesassertRaisesRegexp 已经被重命名为 assertRegex()assertRaisesRegex()

3.5 版後已棄用: assertNotRegexpMatches 这个名称已被弃用并应改用 assertNotRegex()

分组测试

class unittest.TestSuite(tests=())

这个类代表对单独测试用例和测试套件的聚合。 这个类提供给测试运行方所需的接口以允许其像任何其他测试用例一样运行。 运行一个 TestSuite 实例与对套件执行迭代来逐一运行每个测试的效果相同。

如果给出了 tests,则它必须是一个包含单独测试用例的可迭代对象或是将被用于初始构建测试套件的其他测试套件。 还有一些附加的方法会被提供用来在随后向测试集添加测试用例和测试套件。

TestSuite 对象的行为与 TestCase 对象很相似,区别在于它们并不会真正实现一个测试。 它们会被用来将测试聚合为多个要同时运行的测试分组。 还有一些附加的方法会被用来向 TestSuite 实例添加测试:

addTest(test)

向测试套件添加 TestCaseTestSuite

addTests(tests)

将来自包含 TestCaseTestSuite 实例的可迭代对象的所有测试添加到这个测试套件。

这等价于对 tests 进行迭代,并为其中的每个元素调用 addTest()

TestSuiteTestCase 共享下列方法:

run(result)

运行与这个套件相关联的测试,将结果收集到作为 result 传入的测试结果对象中。 请注意与 TestCase.run() 的区别,TestSuite.run() 必须传入结果对象。

debug()

运行与这个套件相关联的测试而不收集结果。 这允许测试所引发的异常被传递给调用方并可被用于支持在调试器中运行测试。

countTestCases()

返回此测试对象所提供的测试数量,包括单独的测试和子套件。

__iter__()

TestSuite 分组的测试总是可以通过迭代来访问。 其子类可以通过重载 __iter__() 来惰性地提供测试。 请注意此方法可在单个套件上多次被调用(例如在计数测试或相等性比较时),为此在 TestSuite.run() 之前重复迭代所返回的测试对于每次调用迭代都必须相同。 在 TestSuite.run() 之后,调用方不应继续访问此方法所返回的测试,除非调用方使用重载了 TestSuite._removeTestAtIndex() 的子类来保留对测试的引用。

3.2 版更變: 在较早的版本中 TestSuite 会直接访问测试而不是通过迭代,因此只重载 __iter__() 并不足以提供所有测试。

3.4 版更變: 在较早的版本中 TestSuite 会在 TestSuite.run() 之后保留对每个 TestCase 的引用。 其子类可以通过重载 TestSuite._removeTestAtIndex() 来恢复此行为。

TestSuite 对象的典型应用中,run() 方法是由 TestRunner 发起调用而不是由最终用户测试来控制。

加载和运行测试

class unittest.TestLoader

TestLoader 类可被用来基于类和模块创建测试套件。 通常,没有必要创建该类的实例;unittest 模块提供了一个可作为 unittest.defaultTestLoader 共享的实例。 但是,使用子类或实例允许对某些配置属性进行定制。

TestLoader 对象具有下列属性:

errors

由在加载测试期间遇到的非致命错误组成的列表。 在任何时候都不会被加载方重围。 致命错误是通过相关方法引发一个异常来向调用方发出信号的。 非致命错误也是由一个将在运行时引发原始错误的合成测试来提示的。

3.5 版新加入.

TestLoader 对象具有下列方法:

loadTestsFromTestCase(testCaseClass)

返回一个包含在 TestCase 所派生的 testCaseClass 中的所有测试用例的测试套件。

会为每个由 getTestCaseNames() 指明的方法创建一个测试用例实例。 在默认情况下这些都是以 test 开头的方法名称。 如果 getTestCaseNames() 不返回任何方法,但 runTest() 方法已被实现,则会为该方法创建一个单独的测试用例。

loadTestsFromModule(module, pattern=None)

返回包含在给定模块中的所有测试用例的测试套件。 此方法会在 module 中搜索从派生自 TestCase 的类并为该类定义的每个测试方法创建一个类实例。

備註

虽然使用 TestCase 所派生的类的层级结构可以方便地共享配置和辅助函数,但在不打算直接实例化的基类上定义测试方法并不能很好地配合此方法使用。 不过,当配置有差异并且定义在子类当中时这样做还是有用处的。

如果一个模块提供了 load_tests 函数则它将被调用以加载测试。 这允许模块自行定制测试加载过程。 这就称为 load_tests protocolpattern 参数会被作为传给 load_tests 的第三个参数。

3.2 版更變: 添加了对 load_tests 的支持。

3.5 版更變: 未写入文档的非官方 use_load_tests 默认参数已被弃用并忽略,但是它仍然被接受以便向下兼容。 此方法现在还接受一个仅限关键字参数 pattern,它会被作为传给 load_tests 的第三个参数。

loadTestsFromName(name, module=None)

返回由给出了字符串形式规格描述的所有测试用例组成的测试套件。

描述名称 name 是一个“带点号的名称”,它可以被解析为一个模块、一个测试用例类、一个测试用例类内部的测试方法、一个 TestSuite 实例,或者一个返回 TestCaseTestSuite 实例的可调用对象。 这些检查将按在此列出的顺序执行;也就是说,一个可能的测试用例类上的方法将作为“一个测试用例内部的测试方法”而非作为“一个可调用对象”被选定。

举例来说,如果你有一个模块 SampleTests,其中包含一个派生自 TestCase 的类 SampleTestCase,其中包含三个测试方法 (test_one(), test_two()test_three())。 则描述名称 'SampleTests.SampleTestCase' 将使此方法返回一个测试套件,它将运行全部三个测试方法。 使用描述名称 'SampleTests.SampleTestCase.test_two' 将使它返回一个测试套件,它将仅运行 test_two() 测试方法。 描述名称可以指向尚未被导入的模块和包;它们将作为附带影响被导入。

本模块可以选择相对于给定的 module 来解析 name

3.5 版更變: 如果在遍历 name 时发生了 ImportErrorAttributeError 则在运行时引发该错误的合成测试将被返回。 这些错误被包括在由 self.errors 所积累的错误中。

loadTestsFromNames(names, module=None)

类似于 loadTestsFromName(),但是接受一个名称序列而不是单个名称。 返回值是一个测试套件,它支持为每个名称所定义的所有测试。

getTestCaseNames(testCaseClass)

返回由 testCaseClass 中找到的方法名称组成的已排序的序列;这应当是 TestCase 的一个子类。

discover(start_dir, pattern='test*.py', top_level_dir=None)

通过从指定的开始目录向其子目录递归来找出所有测试模块,并返回一个包含该结果的 TestSuite 对象。 只有与 pattern 匹配的测试文件才会被加载。 (使用 shell 风格的模式匹配。) 只有可导入的模块名称(即有效的 Python 标识符)将会被加载。

所有测试模块都必须可以从项目的最高层级上导入。 如果起始目录不是最高层级目录则必须单独指明最高层级目录。

如果导入某个模块失败,比如因为存在语法错误,则会将其记录为单独的错误并将继续查找模块。 如果导入失败是因为引发了 SkipTest,则会将其记录为跳过而不是错误。

如果找到了一个包(即包含名为 __init__.py 的文件的目录),则将在包中查找 load_tests 函数。 如果存在此函数则将对其执行调用 package.load_tests(loader, tests, pattern)。 测试发现操作会确保在执行期间仅检查测试一次,即使 load_tests 函数本身调用了 loader.discover 也是如此。.

如果 load_tests 存在则发现操作 不会 对包执行递归处理,load_tests 将负责加载包中的所有测试。is responsible for loading all tests in the package.

模式特意地不被当作 loader 属性来保存以使包能够自己继续执行发现操作。 top_level_dir 则会被保存以使 load_tests 不需要将此参数传入到 loader.discover()

start_dir 可以是一个带点号的名称或是一个目录。

3.2 版新加入.

3.4 版更變: 在导入时引发 SkipTest 的模块会被记录为跳过,而不是错误。

3.4 版更變: start_dir 可以是一个 命名空间包

3.4 版更變: 路径在被导入之前会先被排序以使得执行顺序保持一致,即使下层文件系统的顺序不是取决于文件名的。

3.5 版更變: 现在 load_tests 会检查已找到的包,无论它们的路径是否与 pattern 匹配,因为包名称是无法与默认的模式匹配的。

TestLoader 的下列属性可通过子类化或在实例上赋值来配置:

testMethodPrefix

给出将被解读为测试方法的方法名称的前缀的字符串。 默认值为 'test'

这会影响 getTestCaseNames() 以及所有 loadTestsFrom*() 方法。

sortTestMethodsUsing

将被用来在 getTestCaseNames() 以及所有 loadTestsFrom*() 方法中比较方法名称以便对它们进行排序。

suiteClass

根据一个测试列表来构造测试套件的可调用对象。 不需要结果对象上的任何方法。 默认值为 TestSuite 类。

这会影响所有 loadTestsFrom*() 方法。

testNamePatterns

由 Unix shell 风格通配符的测试名称模式组成的列表,供测试方法进行匹配以包括在测试套件中 (参见 -k 选项)。

如果该属性不为 None (默认值),则将要包括在测试套件中的所有测试方法都必须匹配该列表中的某个模式。 请注意匹配总是使用 fnmatch.fnmatchcase(),因此不同于传给 -k 选项的模式,简单的子字符串模式将必须使用 * 通配符来进行转换。

这会影响所有 loadTestsFrom*() 方法。

3.7 版新加入.

class unittest.TestResult

这个类被用于编译有关哪些测试执行成功而哪些失败的信息。

存放一组测试的结果的 TestResult 对象。 TestCaseTestSuite 类将确保结果被正确地记录;测试创建者无须担心如何记录测试的结果。

建立在 unittest 之上的测试框架可能会想要访问通过运行一组测试所产生的 TestResult 对象用来报告信息;TestRunner.run() 方法是出于这个目的而返回 TestResult 实例的。

TestResult 实例具有下列属性,在检查运行一组测试的结果的时候很有用处。

errors

一个包含 TestCase 实例和保存了格式化回溯信息的字符串 2 元组的列表。 每个元组代表一个引发了非预期的异常的测试。

failures

一个包含 TestCase 实例和保存了格式化回溯信息的字符串 2 元组的列表。 每个元组代表一个使用 TestCase.assert*() 方法显式地发出失败信号的测试。

skipped

一个包含 2-tuples of TestCase 实例和保存了跳过测试原因的字符串 2 元组的列表。

3.1 版新加入.

expectedFailures

一个包含 TestCase 实例和保存了格式化回溯信息的 2 元组的列表。 每个元组代表测试用例的一个已预期的失败或错误。

unexpectedSuccesses

一个包含被标记为已预期失败,但却测试成功的 TestCase 实例的列表。

shouldStop

当测试的执行应当被 stop() 停止时则设为 True

testsRun

目前已运行的测试的总数量。

buffer

如果设为真值,sys.stdoutsys.stderr 将在 startTest()stopTest() 被调用之间被缓冲。 被收集的输出将仅在测试失败或发生错误时才会被回显到真正的 sys.stdoutsys.stderr。 任何输出还会被附加到失败/错误消息中。

3.2 版新加入.

failfast

如果设为真值则 stop() 将在首次失败或错误时被调用,停止测试运行。

3.2 版新加入.

tb_locals

如果设为真值则局部变量将被显示在回溯信息中。

3.5 版新加入.

wasSuccessful()

如果当前所有测试都已通过则返回 True,否则返回 False

3.4 版更變: 如果有任何来自测试的 unexpectedSuccessesexpectedFailure() 装饰器所标记则返回 False

stop()

此方法可被调用以提示正在运行的测试集要将 shouldStop 属性设为 True 来表示其应当被中止。 TestRunner 对象应当认同此旗标并返回而不再运行任何额外的测试。

例如,该特性会被 TextTestRunner 类用来在当用户从键盘发出一个中断信号时停止测试框架。 提供了 TestRunner 实现的交互式工具也可通过类似方式来使用该特性。

TestResult 类的下列方法被用于维护内部数据结构,并可在子类中被扩展以支持额外的报告需求。 这特别适用于构建支持在运行测试时提供交互式报告的工具。

startTest(test)

当测试用例 test 即将运行时被调用。

stopTest(test)

在测试用例 test 已经执行后被调用,无论其结果如何。

startTestRun()

在任何测试被执行之前被调用一次。

3.1 版新加入.

stopTestRun()

在所有测试被执行之后被调用一次。

3.1 版新加入.

addError(test, err)

当测试用例 test 引发了非预期的异常时将被调用。 err 是一个元组,其形式与 sys.exc_info() 的返回值相同: (type, value, traceback)

默认实现会将一个元组 (test, formatted_err) 添加到实例的 errors 属性,其中 formatted_err 是派生自 err 的已格式化回溯信息。

addFailure(test, err)

当测试用例 test 发出了失败信号时将被调用。 err 是一个元组,其形式与 sys.exc_info() 的返回值相同: (type, value, traceback)

默认实现会将一个元组 (test, formatted_err) 添加到实例的 failures 属性,其中 formatted_err 是派生自 err 的已格式化回溯信息。

addSuccess(test)

当测试用例 test 成功时被调用。

默认实现将不做任何操作。

addSkip(test, reason)

当测试用例 test 被跳过时将被调用。 reason 是给出的跳过测试的理由。

默认实现会将一个元组 (test, reason) 添加到实例的 skipped 属性。

addExpectedFailure(test, err)

当测试用例 test 失败或发生错误,但是使用了 expectedFailure() 装饰器来标记时将被调用。

默认实现会将一个元组 (test, formatted_err) 添加到实例的 expectedFailures 属性,其中 formatted_err 是派生自 err 的已格式化回溯信息。

addUnexpectedSuccess(test)

当测试用例 test 使用了was marked with the expectedFailure() 装饰器来标记,但是却执行成功时将被调用。

默认实现会将该测试添加到实例的 unexpectedSuccesses 属性。

addSubTest(test, subtest, outcome)

当一个子测试结束时将被调用。 test 是对应于该测试方法的测试用例。 subtest 是一个描述该子测试的 TestCase 实例。

如果 outcomeNone,则该子测试执行成功。 否则,它将失败并引发一个异常,outcome 是一个元组,其形式与 sys.exc_info() 的返回值相同: (type, value, traceback)

默认实现在测试结果为成功时将不做任何事,并会将子测试的失败记录为普通的失败。

3.4 版新加入.

class unittest.TextTestResult(stream, descriptions, verbosity)

TestResult 的一个具体实现,由 TextTestRunner 使用。

3.2 版新加入: 这个类在之前被命名为 _TextTestResult。 这个旧名字仍然作为别名存在,但已被弃用。

unittest.defaultTestLoader

用于分享的 TestLoader 类实例。 如果不需要自制 TestLoader,则可以使用该实例而不必重复创建新的实例。

class unittest.TextTestRunner(stream=None, descriptions=True, verbosity=1, failfast=False, buffer=False, resultclass=None, warnings=None, *, tb_locals=False)

一个将结果输出到流的基本测试运行器。 如果 stream 为默认的 None,则会使用 sys.stderr 作为输出流。 这个类具有一些配置形参,但实际上都非常简单。 运行测试套件的图形化应用程序应当提供替代实现。 这样的实现应当在添加新特性到 unittest 时接受 **kwargs 作为修改构造运行器的接口。

在默认情况下这个运行器会显示 DeprecationWarning, PendingDeprecationWarning, ResourceWarningImportWarning,即使它们 被默认忽略。 由 deprecated unittest methods 所导致的弃用警告也会被作为特例,并且当警告过滤器为 'default''always' 时,对于每个模块它们将仅显示一次,以避免过多的警告消息。 此种行为可使用 Python 的 -Wd-Wa 选项 (参见 警告控制) 并让 warnings 保持为 None 来覆盖。

3.2 版更變: 新增 warnings 引數。

3.2 版更變: 默认流会在实例化而不是在导入时被设为 sys.stderr

3.5 版更變: 新增 tb_locals 參數。

_makeResult()

此方法将返回由 run() 使用的 TestResult 实例。 它不应当被直接调用,但可在子类中被重载以提供自定义的 TestResult

_makeResult() 会实例化传给 TextTestRunner 构造器的 resultclass 参数所指定的类或可迭代对象。 如果没有提供 resultclass 则默认为 TextTestResult。 结果类会使用以下参数来实例化:

stream, descriptions, verbosity
run(test)

此方法是 TextTestRunner 的主要公共接口。 此方法接受一个 TestSuiteTestCase 实例。 通过调用 _makeResult() 创建 TestResult 来运行测试并将结果打印到标准输出。

unittest.main(module='__main__', defaultTest=None, argv=None, testRunner=None, testLoader=unittest.defaultTestLoader, exit=True, verbosity=1, failfast=None, catchbreak=None, buffer=None, warnings=None)

module 加载一组测试并运行它们的命令行程序;这主要是为了让测试模块能方便地执行。 此函数的最简单用法是在测试脚本末尾包括下列行:

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

你可以通过传入冗余参数运行测试以获得更详细的信息:

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

defaultTest 参数是要运行的单个测试名称,或者如果未通过 argv 指定任何测试名称则是包含多个测试名称的可迭代对象。 如果未指定或为 None 且未通过 argv 指定任何测试名称,则会运行在 module 中找到的所有测试。

argv 参数可以是传给程序的选项列表,其中第一个元素是程序名。 如未指定或为 None,则会使用 sys.argv 的值。

testRunner 参数可以是测试运行器类或是其已创建的实例。 在默认情况下 main 会调用 sys.exit() 并附带一个退出码来指明测试运行是成功还是失败。

testLoader 参数必须是一个 TestLoader 实例,其默认值为 defaultTestLoader

main 支持通过传入 exit=False 参数以便在交互式解释器中使用。 这将在标准输出中显示结果而不调用 sys.exit():

>>> from unittest import main
>>> main(module='test_module', exit=False)

failfast, catchbreakbuffer 形参的效果与同名的 command-line options 一致。

warnings 参数指定在运行测试时所应使用的 警告过滤器。 如果未指定,则默认的 None 会在将 -W 选项传给 python 命令时被保留 (参见 警告控制),而在其他情况下将被设为 'default'

调用 main 实际上将返回一个 TestProgram 类的实例。 这会把测试运行结果保存为 result 属性。

3.1 版更變: 新增 exit 參數。

3.2 版更變: 增加了 verbosity, failfast, catchbreak, bufferwarnings 形参。

3.4 版更變: defaultTest 形参被修改为也接受一个由测试名称组成的迭代器。

load_tests 协议

3.2 版新加入.

模块或包可以通过实现一个名为 load_tests 的函数来定制在正常测试运行或测试发现期间要如何从中加载测试。

如果一个测试模块定义了 load_tests 则它将被 TestLoader.loadTestsFromModule() 调用并传入下列参数:

load_tests(loader, standard_tests, pattern)

其中 pattern 会通过 loadTestsFromModule 传入。 它的默认值为 None

它应当返回一个 TestSuite

loader 是执行载入操作的 TestLoader 实例。 standard_tests 是默认要从该模块载入的测试。 测试模块通常只需从标准测试集中添加或移除测试。 第三个参数是在作为测试发现的一部分载入包时使用的。

一个从指定 TestCase 类集合中载入测试的 load_tests 函数看起来可能是这样的:

test_cases = (TestCase1, TestCase2, TestCase3)

def load_tests(loader, tests, pattern):
    suite = TestSuite()
    for test_class in test_cases:
        tests = loader.loadTestsFromTestCase(test_class)
        suite.addTests(tests)
    return suite

如果发现操作是在一个包含包的目录中开始的,不论是通过命令行还是通过调用 TestLoader.discover(),则将在包 __init__.py 中检查 load_tests。 如果不存在此函数,则发现将在包内部执行递归,就像它是另一个目录一样。 在其他情况下,包中测试的发现操作将留给 load_tests 执行,它将附带下列参数被调用:

load_tests(loader, standard_tests, pattern)

这应当返回代表包中所有测试的 TestSuite。 (standard_tests 将只包含从 __init__.py 获取的测试。)

因为模式已被传入 load_tests 所以包可以自由地继续(还可能修改)测试发现操作。 针对一个测试包的 '无操作' load_tests 函数看起来是这样的:

def load_tests(loader, standard_tests, pattern):
    # top level directory cached on loader instance
    this_dir = os.path.dirname(__file__)
    package_tests = loader.discover(start_dir=this_dir, pattern=pattern)
    standard_tests.addTests(package_tests)
    return standard_tests

3.5 版更變: 发现操作不会再检查包名称是否匹配 pattern,因为包名称不可能匹配默认的模式。

类与模块设定

类与模块设定是在 TestSuite 中实现的。 当测试套件遇到来自新类的测试时则来自之前的类(如果存在)的 tearDownClass() 会被调用,然后再调用来自新类的 setUpClass()

类似地如果测试是来自之前的测试的另一个模块则来自之前模块的 tearDownModule 将被运行,然后再运行来自新模块的 setUpModule

在所有测试运行完毕后最终的 tearDownClasstearDownModule 将被运行。

请注意共享设定不适用于一些 [潜在的] 特性例如测试并行化并且它们会破坏测试隔离。 它们应当被谨慎地使用。

由 unittest 测试加载器创建的测试的默认顺序是将所有来自相同模块和类的测试归入相同分组。 这将导致 setUpClass / setUpModule (等) 对于每个类和模块都恰好被调用一次。 如果你将顺序随机化,以便使得来自不同模块和类的测试彼此相邻,那么这些共享的设定函数就可能会在一次测试运行中被多次调用。

共享的设定不适用与非标准顺序的套件。 对于不想支持共享设定的框架来说 BaseTestSuite 仍然可用。

如果在共享的设定函数中引发了任何异常则测试将被报告错误。 因为没有对应的测试实例,所以会创建一个 _ErrorHolder 对象(它具有与 TestCase 相同的接口)来代表该错误。 如果你只是使用标准 unittest 测试运行器那么这个细节并不重要,但是如果你是一个框架开发者那么这可能会有关系。

setUpClass 和 tearDownClass

这些必须被实现为类方法:

import unittest

class Test(unittest.TestCase):
    @classmethod
    def setUpClass(cls):
        cls._connection = createExpensiveConnectionObject()

    @classmethod
    def tearDownClass(cls):
        cls._connection.destroy()

如果你希望在基类上的 setUpClasstearDownClass 被调用则你必须自己去调用它们。 在 TestCase 中的实现是空的。

如果在 setUpClass 中引发了异常则类中的测试将不会被运行并且 tearDownClass 也不会被运行。 跳过的类中的 setUpClasstearDownClass 将不会被运行。 如果引发的异常是 SkipTest 异常则类将被报告为已跳过而非发生错误。

setUpModule 和 tearDownModule

这些应当被实现为函数:

def setUpModule():
    createConnection()

def tearDownModule():
    closeConnection()

如果在 setUpModule 中引发了异常则模块中的任何测试都将不会被运行并且 tearDownModule 也不会被运行。 如果引发的异常是 SkipTest 异常则模块将被报告为已跳过而非发生错误。

要添加即使在发生异常时也必须运行的清理代码,请使用 addModuleCleanup:

unittest.addModuleCleanup(function, /, *args, **kwargs)

tearDownModule() 之后添加一个要调用的函数来清理测试类运行期间所使用的资源。 函数将按它们被添加的相反顺序被调用 (LIFO)。 它们在调用时将附带它们被添加时传给 addModuleCleanup() 的任何参数和关键字参数。

如果 setUpModule() 失败,即意味着 tearDownModule() 未被调用,则已添加的任何清理函数仍将被调用。

3.8 版新加入.

unittest.doModuleCleanups()

此函数会在 tearDownModule() 之后无条件地被调用,或者如果 setUpModule() 引发了异常则会在 setUpModule() 之后被调用。

它将负责调用由It is responsible for calling all the cleanup functions added by addModuleCleanup() 添加的所有清理函数。 如果你需要在 tearDownModule() 之前 调用清理函数则可以自行调用 doModuleCleanups()

doModuleCleanups() 每次会弹出清理函数栈中的一个方法,因此它可以在任何时候被调用。

3.8 版新加入.

信号处理

3.2 版新加入.

unittest 的 -c/--catch 命令行选项,加上 unittest.main()catchbreak 形参,提供了在测试运行期间处理 control-C 的更友好方式。 在捕获中断行为被启用时 control-C 将允许当前运行的测试能够完成,而测试运行将随后结束并报告已有的全部结果。 第二个 control-C 将会正常地引发 KeyboardInterrupt

处理 control-C 信号的处理器会尝试与安装了自定义 signal.SIGINT 处理器的测试代码保持兼容。 如果是 unittest 处理器而 不是 已安装的 signal.SIGINT 处理器被调用,即它被系统在测试的下层替换并委托处理,则它会调用默认的处理器。 这通常会是替换了已安装处理器并委托处理的代码所预期的行为。 对于需要禁用 unittest control-C 处理的单个测试则可以使用 removeHandler() 装饰器。

还有一些工具函数让框架开发者可以在测试框架内部启用 control-C 处理功能。

unittest.installHandler()

安装 control-C 处理器。 当接收到 signal.SIGINT 时(通常是响应用户按下 control-C)所有已注册的结果都会执行 stop() 调用。

unittest.registerResult(result)

注册一个 TestResult 对象用于 control-C 的处理。 注册一个结果将保存指向它的弱引用,因此这并不能防止结果被作为垃圾回收。

如果 control-C 未被启用则注册 TestResult 对象将没有任何附带影响,因此不论是否启用了该项处理测试框架都可以无条件地注册他们独立创建的所有结果。

unittest.removeResult(result)

移除一个已注册的结果。 一旦结果被移除则 stop() 将不再会作为针对 control-C 的响应在结果对象上被调用。

unittest.removeHandler(function=None)

当不附带任何参数被调用时此函数将移除已被安装的 control-C 处理器。 此函数还可被用作测试装饰器以在测试被执行时临时性地移除处理器:

@unittest.removeHandler
def test_signal_handling(self):
    ...