Подтвердить что ты не робот

Любой способ reset издеваться над исходным состоянием? - Python Mock - mock 1.0b1

У меня есть следующий упрощенный класс, который я издеваюсь:

class myClass(object):
    @staticmethod
    def A():
        #...

    def check(self):
        #code...
        value = self.A()
        #more code...

В моем первом тесте я высмеиваю только метод A

from django.test import TestCase
from mock import MagicMock
import myClass

class FirstTest(TestCase):

def setUp(self):
    myClass.A = MagicMock(return_value = 'CPU')

def test(self):
    #some tests 
    myClassObj = myClass()
    myClassObj.check()

В то время как в моем втором тесте я высмеиваю весь метод проверки:

from django.test import TestCase
from mock import MagicMock
import myClass

class SecondTest(TestCase):

def setUp(self):
    myClass.check = MagicMock(return_value = someObject)

def test(self):
    #some tests 
    myClassObj = myClass()
    myClassObj.check()

Теперь мои утверждения из моего первого теста терпят неудачу, потому что вместо вызова check() и издевки A() внутри check() он вызывает полностью издеваемое check() из моего второго теста.

Есть ли способ очистить и установить метод "нормальный" после теста? Я уже пробовал myClass.check.reset_mock(), но он ничего не делает. Перемещение порядка моих тестов тоже не делает.

Я использую mock 1.0b1 для python из http://pypi.python.org/pypi/mock/

4b9b3361

Ответ 1

Вы можете спрятать функцию самостоятельно и вернуть ее, когда закончите.

import unittest

from mock import MagicMock
from MyClass import MyClass

class FirstTest(unittest.TestCase):

    def setUp(self):
        self.A = MyClass.A
        MyClass.A = MagicMock(name='mocked A', return_value='CPU')


    def tearDown(self):
        MyClass.A = self.A

    def test_mocked_static_method(self):
        print 'First Test'
        print MyClass.check
        print MyClass.A


class SecondTest(unittest.TestCase):

    def setUp(self):
        MyClass.check = MagicMock(name='mocked check', return_value=object)

    def test_check_mocked_check_method(self):
        print 'Second Test'
        print MyClass.check
        print MyClass.A


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

Запуск этого файла дает следующий результат:

First Test
<unbound method MyClass.check> 
<MagicMock name='mocked A' id='141382732'>
Second Test
<MagicMock name='mocked check' id='141382860'>
<unbound method MyClass.A>

Я нашел себя с помощью декоратора патча намного больше, чем setUp и tearDown сейчас. В этом случае вы могли бы сделать

from mock import patch

@patch('MyClass.A')
def test_mocked_static_method(self, mocked_A)
    mocked_A.return_value = 'CPU'
    # This mock will expire when the test method is finished

Ответ 2

Вы можете использовать mock.patch в качестве декоратора или менеджера контекста:

from mock import patch, MagicMock

@patch('myClass.A', MagicMock(return_value='CPU'))
def test(self):
    pass

или

def test(self):
    with patch('myClass.A', MagicMock(return_value='CPU')):
        pass

Если вы не даете макету объекта patch, то он предоставит автоспекутированный макет, который вы можете изменить:

@patch('myClass.A')
def test(self, mock_A):
    mock_A.return_value = 'CPU'
    pass

или

def test(self):
    with patch('myClass.A') as mock_A:
        mock_A.return_value = 'CPU'
        pass

Во всех случаях исходное значение будет восстановлено после завершения декорированной тестовой функции или менеджера контекста.