使用 nose 進(jìn)行自動化測試

作者:廖飛

全文約 7778 字,讀完可能需要 15 分鐘。

原文鏈接:http://www.cnblogs.com/liaofeifight/

一、為什么使用 nose ?

  1. 編寫測試更容易。

    nose 可以自動識別繼承于 unittest.TestCase 的測試單元,并執(zhí)行測試,而且, nose 也可以測試非繼承于 unittest.TestCase 的測試單元。 nose 提供了豐富的 API 便于編寫測試代碼。


  2. 執(zhí)行測試更容易。

    只要遵循一些簡單的規(guī)則去組織你的類庫和測試代碼, nose 是可以自動識別單元測試的。執(zhí)行測試是非常耗資源的,但是,一段第一個測試模塊被加載后, nose 就開始執(zhí)行測試。


  3. 建立測試環(huán)境更容易。

  4. 做你想做的事情更容易。


    nose 擁有很多內(nèi)置的插件幫助進(jìn)行輸出抓取、錯誤查找、代碼覆蓋、文檔測試( doctest )等等。同樣,如果你不喜歡這些內(nèi)置插件提供的功能或者這些插件不能滿足你的項目結(jié)構(gòu),你可以自定義開發(fā)插件來完成你想要做的事情。

二、基本使用

安裝: pip install ?nose

腳本命令:

nosetests [options] ?[( optional) test files ?or directories]

除了使用命令行這種方式之外,還可以在根目錄下放置配置文件,配置文件的類型為 .noserc 或 nose.cfg 文件。配置文件都是標(biāo)準(zhǔn)的 ini 內(nèi)容格式。例如:

三、擴展使用

nose 自動收集單元測試,收集它當(dāng)前工作目錄下的源代碼文件、目錄以及包。任何的源代碼文件、目錄或者包只要匹配正則表達(dá)式,他們就會被自動收集。包的測試收集按照樹的層級級別一級一級進(jìn)行,因此 package.tests 、 package.sub.tests 、 package.sub.sub2.tests 將會被收集。

匹配成功的包、任何 python 的源文件都會當(dāng)做測試用例。

3.1 選擇測試用例

將需要測試的名稱傳遞給 nose 的命令行。格式如下:

nosetests only_test_this.py

測試的名稱可以是腳本文件的名稱或者模塊的名稱,也可以使用 colon 表達(dá)式表達(dá)的測試名稱。路徑可以是相對的路徑也可以是絕對的路徑。如下所示:

同樣可以使用—— w 開關(guān)來切換當(dāng)前的工作路徑,從而改變 nose 查找測試用例的根路徑。用法如下:

nosetests -w /path/to/tests

更多關(guān)于自定義測試用例的收集與加載方式,可以使用插件的方式做到。

3.2 配置

除了3.1通過腳本命令傳遞參數(shù)的方式外,你還可以在根目錄下通過設(shè)置 setup.cfg 或者 .noserc 或者 nose.cfg 等配置文件達(dá)到同樣的目的。例如:

所有查找到的配置文件將會被加載,而且配置項的值會合并。如果想覆蓋標(biāo)準(zhǔn)的配置文件,使用—— c 選項。

3.3 使用插件

使用 pip 安裝所需要的插件,然后通過 nosetests 命令行配置插件。執(zhí)行如下命令驗證所安裝的插件:

nosetests - plugins

在命令行中添加 -v 或者 -vv 選項可以顯示每一個插件的更多信息。?如果通過 nose.main() 或者 nose.run() 執(zhí)行測試,可以將要使用的插件關(guān)鍵字參數(shù)的列表傳遞進(jìn)去。

3.4 配置選項

  • V ,— version

輸出 nose 的版本

  • p ,— plugins

輸出可獲取的插件列表。

  • v = DEFUALT ,— verbose = DEFAULT

使用更多的 verbose

— verbosity = VERBOSITY

設(shè)置 verbosity ;— verbosity =2與—— v 設(shè)置一致

  • q ,— quiet = DEFAULT

使用更少的 verbose

  • c = FILES ,— config = FILES

設(shè)置配置文件??梢栽O(shè)置很多次,然后將所有的配置文件合并。

  • w = WHERE ,— where = WHERE

設(shè)置查找的根目錄。

  • py 3 where = WHERE

顧名思義,針對 python 3. x 以上設(shè)置查找路徑。

  • m = REGEX ,— match = REGEX ,— testmatch = REGEX

設(shè)置用于自動化收集用例的正則表達(dá)式。

— tests = NAMES

執(zhí)行這些測試。

— debug - log = FILE

設(shè)置調(diào)試的日志文件路徑。

— logging - config = FILE ,— log - config = FILE

設(shè)置日志文件的配置文件。

  • I = REGEX ,— ignore - files = REGEX

設(shè)置自動收集測試用例時忽略的正則表達(dá)式。

  • e = REGEX ,— exclude = REGEX

排除要執(zhí)行的測試用例的正則表達(dá)式

  • i = REGEX ,— include = REGEX

包含要執(zhí)行的測試用例的正則表達(dá)式

  • x ,— stop

執(zhí)行測試發(fā)生錯誤后,停止執(zhí)行測試。

— noexe

不查找可以執(zhí)行文件。

  • a = ATTR ,— attr = ATTR

只執(zhí)行包含 ATTR 屬性的測試用例。

  • A = EXPR ,— eval - attr = EXPR

只執(zhí)行屬性與 EXPR 匹配的測試用例。

  • s ,— nocapture

不抓取標(biāo)準(zhǔn)輸出 \ ( stdout\ )

— nologcapture

禁止使用日志插件

— logging - format = FORMAT

打印語句的自定義格式

— logging - datefmt = FORMAT

日志的日期時間格式

— logging - filter = FILTER

日志語句的過濾器。

— logging - clear - handlers

清除日志的其他 handlers

— logging - level = DEFUALT

設(shè)置日志的等級

— with - coverage

開啟 coverage 插件

— cover - package = PACKAGE

限定 coverage 所在包

— cover - erase

在執(zhí)行之前 清除上次 coverage 統(tǒng)計結(jié)果

— cover - testes

在 coverage 報告中包含測試模塊

— cover - html

產(chǎn)生 html 的 coverage 報告

— cover - html - dir = DIR

設(shè)置存儲 html 的目錄

— cover - xml

產(chǎn)生 xml 的 coverage 報告

— cover - xml - file = FILE

設(shè)置存儲 coverage 報告的 xml 文件

— cover - config - file = DEFUALT

設(shè)置 coverage 的配置文件

  • pdb

當(dāng)測試失敗或產(chǎn)生錯誤是進(jìn)入調(diào)試模式

— pdb - failures

當(dāng)測試失敗時進(jìn)入調(diào)試模式

— pdb - errors

當(dāng)測試產(chǎn)生錯誤時進(jìn)入調(diào)試模式

— with - doctest

開啟 doctest 插件

— doctest - tests

在測試模塊中查詢 doctests

— with - profile

開啟 profile 插件

— profile - sort = SORT

設(shè)置 profiler 輸出排序

— profile - stats - file = FILE

設(shè)置統(tǒng)計所在的文件地址

— with - id

開啟 TestId 插件

— processes = NUM

開始測試處理器的個數(shù)

— processes - timeout = SECONDS

設(shè)置超時時間。

— with - xuint

開始 Xunit 插件

— xuint - file = FILE

設(shè)置 XUnit 報告所在的 xml 文件

— all - modules

開啟 AllModules 插件

— collect - only

開啟只收集測試功能。只收集測試用例及輸出測試名字,而不執(zhí)行測試

四、編寫測試用例

4.1 項目組織結(jié)構(gòu)

由于 nose 是自動收集測試用例的,只有 nose 執(zhí)行的測試目錄下的源代碼文件、包名、子目錄名跟正則表達(dá)式匹配成功后,才能被收集。而且代碼是樹級層次顯示的話, nose 會逐級向下查找子目錄下的匹配的測試用例。

匹配的正則表達(dá)式默認(rèn)值為: ( (?: ^ | [\b_\. - ] ) [Tt]est. 所以最好是以 Test 開頭,或者 test 開頭。當(dāng)然也可以修改默認(rèn)的匹配的正則表達(dá)式。

所以,推薦的項目結(jié)構(gòu)為:

為項目單獨建一個 test 包,里面按項目模塊分子包,最后以及為 "test_" 開頭的測試用例源文件。

4.2 測試結(jié)果

4.3測試代碼

test_mymath.py 中的代碼如下:

# -*- coding: utf-8 -*-

__author__ = 'Administrator'

from nose.tools import assert_equal

from nose.tools import with_setup

import ?unittest

import mymath.math as mmath


def setUp(): ? ?

????print("============test math module setup==============")


def
teardown(): ? ?

? ? print("============test math module teardown==============")


def
test_math_add():

? ? result=mmath.add(4, 5) ? ?

? ? print("================test_math_add============")

? ?assert_equal(10, result)


class test_math3(): ? ?

? ? def setUp(self): ? ? ? ?

? ? ? ? print("============test math class setup==============")
? ??
def teardown(self): ? ? ? ?

????????print("============test math class teardown==============") ? ?

????def test_math_square(self): ? ? ? ?

????????print("=============== test_math_square================ ")

? ? ? ?assert_equal(9, mmath.square(3)) ? ?

????def test_math_sub(self): ? ? ? ?

????????print("=============== test_math_sub================ ")

? ? ? ?assert_equal(1, mmath.sub(3, 2))


class test_math2(unittest.TestCase): ? ?

????def test_math_multipy(self): ? ? ? ?

????????print("=============== test_math_multipy================ ")

? ? ? ?assert_equal(8, mmath.multiply(2, 4))

test_logicOper.py 中的代碼如下:

4.4 測試分析

(1)測試用例的收集

由于 nose 是按照正則表達(dá)式自動收集匹配的測試用例。我們這里收集了5個測試用例。分別了

  • test_mymath.py 模塊里的 test_math_add 函數(shù)。

  • test_mymath.py 模塊里的 test_math 3類下面的 test_math_square 、 test_math_sub 兩個函數(shù)。

  • test_mymath.py 模塊里的 test_math 2類下面的 test_math_multipy 函數(shù)。

  • test_logicOper.py 模塊里的 test_logicOper_isGreaterThan 函數(shù)。

(2)測試的運行順序。

從下圖打印輸出的結(jié)果可以看出。收集的測試用例的執(zhí)行順序:

test_logicOper_isGreaterThan—>test_mymath(模塊).setup—>test_math2(類).test_math_multipy—>test_math3(類).setup—>test_math3(類).test_math_square—>?test_math3(類).teardown—>test_math3(類).setup—>test_math3(類).test_math_sub—>test_math3(類).teardown—>test_mymath(模塊).test_math_add—>test_mymath(模塊).teardown.

大體可以得出如下結(jié)論:

1)測試的順序總體上按照包—>模塊—>類的順序進(jìn)行;

2) 然后按照測試用例名稱的升序排序進(jìn)行,比如test_math_multipy比test_math_square先執(zhí)行;

3)當(dāng)測試模塊中既包含測試函數(shù),又包含測試類時,都一定是先執(zhí)行setup(如果定義了),模塊測試執(zhí)行完畢后執(zhí)行teardown(如果定義了)。而且模塊的setup、setdown只執(zhí)行一次。

4)測試類中的每個測試方法執(zhí)行前先執(zhí)行setup(如果定義了),執(zhí)行完畢后執(zhí)行teardown(如果定義了)。而且每個測試方法的執(zhí)行過程都是如此,新的方法重新按setup—>執(zhí)行方法—>teardown的順序執(zhí)行

(3)關(guān)于 setup 、 teardown

package 、 module 、 class 都可以設(shè)置 setup 、 teardown 。

1) package 的 setup 、 teardown 的設(shè)置

放在 __init__.py 文件中,在整個測試的運行期間只運行一次。

setup 函數(shù)的取名可以是 setup , setup_package , setUp , setUpPackage 中的一個。

teardown 函數(shù)取名可以是 teardown , teardown_package , tearDown or ?tearDownPackage 中的一個。

2) module 的 setup 、 teardown 的設(shè)置

在整個測試的運行期間只運行一次。

setup 函數(shù)的取名可以是 setup , setup_module , setUp or ?setUpModule 中的一個。

teardown 函數(shù)取名可以是 teardown_module , or tearDownModule 中的一個。

3) class 的 setup 、 teardown 的設(shè)置

每個測試方法執(zhí)行時都會調(diào)用。

setup 函數(shù)的取名可以是 setup_class , setupClass , setUpClass , setupAll , setUpAll 中的一個。

teardown 函數(shù)取名可以是 teardown_class , teardownClass , tearDownClass , teardownAll
, tearDownAll 中的一個。

測試類可以繼承 unittest.TestCase ,也可以不繼承, nose 都能識別。

4)測試方法的 setup 、 teardown

可以通過 with_setup 裝飾器進(jìn)行設(shè)置,比如

總之, nose 的口號是: nose extends ?unittest to ?make testing ?easier (測試更簡單 ?).


點擊閱讀原文,查看更多 Python 教程和資源


閱讀原文:http://mp.weixin.qq.com/s?timestamp=1480650997&src=3&ver=1&signature=rAz8xPmr3gir4SClxdIaYaU9jeefMzjQI9GdwDxmonAAP1Eoea3cA91rWu5aMV15KGFJ3H6fNGnWtw-r0F4SUk6atzeq0qBnYmqNC41VuNjwtEX1VMfmlM6P8ITdXP3vCF86PaA8zGXvFu3ncWTd18AyU-*2s3PXarERbqg8pxg=
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

推薦閱讀更多精彩內(nèi)容