Пересмешивание файловых объектов или итераций в python - PullRequest
13 голосов
/ 17 ноября 2011

Какой способ подходит для проверки и проверки кода, который возвращает объект, возвращаемый open () , с использованием библиотеки mock ?

whitelist_data.py:

WHITELIST_FILE = "testdata.txt"

format_str = lambda s: s.rstrip().lstrip('www.')
whitelist = None

with open(WHITELIST_FILE) as whitelist_data:
    whitelist = set(format_str(line) for line in whitelist_data)

if not whitelist:
    raise RuntimeError("Can't read data from %s file" % WHITELIST_FILE)

def is_whitelisted(substr):
    return 1 if format_str(substr) in whitelist else 0

Вот как я пытаюсь это проверить.

import unittest
import mock 

TEST_DATA = """
domain1.com
domain2.com
domain3.com
"""

class TestCheckerFunctions(unittest.TestCase):

    def test_is_whitelisted_method(self):
        open_mock = mock.MagicMock()
        with mock.patch('__builtin__.open',open_mock):
            manager = open_mock.return_value.__enter__.return_value
            manager.__iter__ = lambda s: iter(TEST_DATA.splitlines())
            from whitelist_data import is_whitelisted
            self.assertTrue(is_whitelisted('domain1.com'))

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

Результат python tests.py:

$ python tests.py

E
======================================================================
ERROR: test_is_whitelisted_method (__main__.TestCheckerFunctions)
----------------------------------------------------------------------
Traceback (most recent call last):
  File "tests.py", line 39, in test_is_whitelisted_method
    from whitelist_data import is_whitelisted
  File "/Users/supa/Devel/python/whitelist/whitelist_data.py", line 20, in <module>
    whitelist = set(format_str(line) for line in whitelist_data)
TypeError: 'Mock' object is not iterable

----------------------------------------------------------------------
Ran 1 test in 0.001s

UPD: Спасибо АдамуЯ переустановил макет библиотеки (pip install -e hg+https://code.google.com/p/mock#egg=mock) и обновил tests.py.Работает как шарм.

1 Ответ

16 голосов
/ 17 ноября 2011

Вы ищете MagicMock.Это поддерживает итерацию.

В макете 0.80beta4, patch возвращает MagicMock.Итак, этот простой пример работает:

import mock

def foo():
    for line in open('myfile'):
        print line

@mock.patch('__builtin__.open')
def test_foo(open_mock):
    foo()
    assert open_mock.called

Если вы используете mock 0.7.x (похоже, что вы есть), я не думаю, что вы можете сделать это с помощью одного патча.Вам нужно создать макет отдельно, а затем передать его в патч:

import mock

def foo():
    for line in open('myfile'):
        print line

def test_foo():
    open_mock = mock.MagicMock()
    with mock.patch('__builtin__.open', open_mock):
        foo()
        assert open_mock.called

Примечание. Я запустил их с py.test, однако эти же подходы будут работать с unittest какхорошо.

...