python unittest howto

时间:2010-04-16 19:41:20

标签: python unit-testing

我想知道如何对以下模块进行单元测试。

def download_distribution(url, tempdir):
    """ Method which downloads the distribution from PyPI """
    print "Attempting to download from %s" % (url,)

    try:
        url_handler = urllib2.urlopen(url)
        distribution_contents = url_handler.read()
        url_handler.close()

        filename = get_file_name(url)

        file_handler = open(os.path.join(tempdir, filename), "w")
        file_handler.write(distribution_contents)
        file_handler.close()
        return True

    except ValueError, IOError:
        return False

3 个答案:

答案 0 :(得分:5)

模糊的问题。如果你只是在寻找一般的单元测试的底漆,我会推荐Mark Pilgrim的“Dive Into Python”,它有a chapter on unit testing with Python。否则,您需要清除测试该代码的具体问题。

答案 1 :(得分:5)

单元测试命题器会告诉您单元测试应该是自包含的,也就是说,它们不应该访问网络或文件系统(特别是在写入模式下)。网络和文件系统测试超出了单元测试的范围(尽管您可能会对它们进行集成测试)。

一般来说,对于这种情况,我会将urllib和文件编写代码提取到单独的函数(不进行单元测试),并在单元测试期间注入模拟函数。

即。 (稍微缩写以便更好地阅读):

def get_web_content(url):
    # Extracted code
    url_handler = urllib2.urlopen(url)
    content = url_handler.read()
    url_handler.close()
    return content

def write_to_file(content, filename, tmpdir):
    # Extracted code
    file_handler = open(os.path.join(tempdir, filename), "w")
    file_handler.write(content)
    file_handler.close()

def download_distribution(url, tempdir):
    # Original code, after extractions
    distribution_contents = get_web_content(url)
    filename = get_file_name(url)
    write_to_file(distribution_contents, filename, tmpdir)
    return True

并且,在测试文件上:

import module_I_want_to_test

def mock_web_content(url):
    return """Some fake content, useful for testing"""
def mock_write_to_file(content, filename, tmpdir):
    # In this case, do nothing, as we don't do filesystem meddling while unit testing
    pass

module_I_want_to_test.get_web_content = mock_web_content
module_I_want_to_test.write_to_file = mock_write_to_file

class SomeTests(unittest.Testcase):
    # And so on...

然后我提出丹尼尔的建议,你应该阅读更深入的单元测试资料。

答案 2 :(得分:0)

要模拟urllopen,您可以预先获取一些可以在单元测试中使用的示例。这是一个让你入门的例子:

def urlopen(url):
    urlclean = url[:url.find('?')] # ignore GET parameters
    files = {
        'http://example.com/foo.xml': 'foo.xml',
        'http://example.com/bar.xml': 'bar.xml',
    }
    return file(files[urlclean])
yourmodule.urllib.urlopen = urlopen