深度剖析Python面向?qū)ο笳Z(yǔ)言內(nèi)容
Python面向?qū)ο笳Z(yǔ)言必須執(zhí)行一些復(fù)雜的操作,比如在 Firefox 中打開網(wǎng)頁(yè)并單擊 “Continue” 按鈕,然后檢查結(jié)果。在開始實(shí)際測(cè)試(比如打開頁(yè)面并單擊按鈕)之前,測(cè)試必須先完成一些步驟。
現(xiàn)在,考慮如果一百個(gè)功能性測(cè)試都要這樣執(zhí)行測(cè)試,會(huì)怎么樣。它們都需要通過調(diào)用共同的 setup 例程運(yùn)行 Firefox。然后才能執(zhí)行自己的測(cè)試。與此相應(yīng),為了取消 setup 所做的操作,可能還有 teardown 代碼。這樣,在測(cè)試套件中就會(huì)增加兩百個(gè)額外的函數(shù)調(diào)用。每個(gè)測(cè)試函數(shù)會(huì)像下面這樣:
- # Naming a with_setup decorator
- firefox_test = with_setup(firefox_setup, firefox_teardown)
- @firefox_test
- def test_index_click():
- ...
- @firefox_test
- def test_index_menu():
- ...
為了消除這些重復(fù)的代碼,許多測(cè)試框架提供了一次性指定每個(gè)測(cè)試都需要運(yùn)行的 setup 和 teardown 代碼的機(jī)制。本文討論的三種框架 zope.testing、py.test 和 nose 都支持程序員編寫的 unittest.TestCase 類中的標(biāo)準(zhǔn) setUp() 和 tearDown() 例程。
但是,除此之外,Python面向?qū)ο笳Z(yǔ)言各個(gè)框架為共同 setup 代碼提供的特性有顯著差異。zope.testing 本身沒有為 setup 和 teardown 提供額外支持。但是前面討論過的 z3c.testsetup 擴(kuò)展會(huì)對(duì) doctest 做一些有意思的處理。
它通過在文件中尋找 :Test-Layer: 字符串來(lái)尋找測(cè)試。doctest 中的層實(shí)際上可以指定兩個(gè)值之一。如果把 doctest 標(biāo)為屬于 unit 層,就意味著運(yùn)行它不需要任何特殊的 setup。但是,如果把它標(biāo)為屬于 functional 層,就意味著只能在調(diào)用框架 setup 函數(shù)之后運(yùn)行它。
通常情況下,:Test-Layer: functional 測(cè)試被設(shè)計(jì)為在完整地配置了 Zope Web 框架的情況下運(yùn)行。因此它們可以創(chuàng)建測(cè)試瀏覽器實(shí)例、發(fā)送請(qǐng)求和查看 Web 框架返回的響應(yīng)。通過代表 doctest 執(zhí)行 setup,z3c.testsetup 可以避免在每個(gè)功能性 doctest 中復(fù)制大量樣板代碼。
最后一項(xiàng)減少樣板代碼的便捷特性是,可以向 z3c.testsetup 提供一個(gè)預(yù)裝載到每個(gè)單元 doctest 的名稱空間中的變量列表。以及另一個(gè)預(yù)裝載到每個(gè)功能性 doctest 中的變量列表。這樣就不需要在每個(gè) doctest 文件的開頭復(fù)制一組相同的 import 語(yǔ)句。
Python面向?qū)ο笳Z(yǔ)言 在默認(rèn)情況下不提供對(duì) setup 和 teardown 的支持。它甚至不運(yùn)行標(biāo)準(zhǔn) unittest.TestCase 類的 setUp() 和 tearDown() 方法,除非打開它的 unittest 插件。nose 在支持共同測(cè)試代碼方面是最出色的。在尋找測(cè)試時(shí),nose 跟蹤記錄找到測(cè)試的上下文。它認(rèn)為 unittest.TestCase 子類內(nèi)部的每個(gè)測(cè)試方法是這個(gè)類 “內(nèi)部的”。
因此由它的 setUp() 和 tearDown() 方法控制,它還認(rèn)為測(cè)試存在于它們的模塊、包含模塊的包以及外層所有包的 “內(nèi)部”。因此,對(duì)于多層 “同心” 容器內(nèi)的測(cè)試,nose 會(huì)在運(yùn)行測(cè)試之前運(yùn)行所有容器中的 setup 代碼,在運(yùn)行測(cè)試之后運(yùn)行所有容器中的 teardown 代碼。
最后,在 @with_setup 修飾符中指定的或作為 unittest.TestCase 子類中的方法提供的 setup 和 teardown 函數(shù)對(duì)于相關(guān)的每個(gè)函數(shù)或測(cè)試運(yùn)行一次,而在模塊級(jí)或包級(jí)向 nose 提供的 setup 和 teardown 代碼對(duì)于整個(gè)測(cè)試集只運(yùn)行一次。因此,不要認(rèn)為這樣的測(cè)試是完全互相隔離的:它們會(huì)共享在模塊或包的 setup 例程中創(chuàng)建的資源拷貝。
【編輯推薦】