Утверждать, что метод вызывался в модульном тесте Python - PullRequest
66 голосов
/ 30 сентября 2010

Предположим, у меня есть следующий код в модульном тесте Python:

aw = aps.Request("nv1")
aw2 = aps.Request("nv2", aw)

Есть ли простой способ утверждать, что определенный метод (в моем случае aw.Clear()) был вызван во второй строке теста? например есть ли что-то вроде этого:

#pseudocode:
assertMethodIsCalled(aw.Clear, lambda: aps.Request("nv2", aw))

Ответы [ 5 ]

111 голосов
/ 19 января 2011

Я использую Mock (который теперь unittest.mock на py3.3 +) для этого:

from mock import patch
from PyQt4 import Qt


@patch.object(Qt.QMessageBox, 'aboutQt')
def testShowAboutQt(self, mock):
    self.win.actionAboutQt.trigger()
    self.assertTrue(mock.called)

Для вашего случая это может выглядетьнапример:

import mock
from mock import patch


def testClearWasCalled(self):
   aw = aps.Request("nv1")
   with patch.object(aw, 'Clear') as mock:
       aw2 = aps.Request("nv2", aw)

   mock.assert_called_with(42) # or mock.assert_called_once_with(42)

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

Caveat emptor! (Покупатель остерегается!)

Если вы наберете опечатку assert_called_with (до assert_called_once или assert_called_wiht), ваш тест все еще может быть запущен, как будет делать Mockдумайте, что это насмешливая функция, и радуйтесь, если вы не используете autospec=true.Для получения дополнительной информации прочитайте assert_called_once: угроза или угроза .

21 голосов
/ 08 августа 2016

Да, если вы используете Python 3.3+.Вы можете использовать встроенный unittest.mock для подтверждения вызванного метода.Для Python 2.6+ используйте скользящий обратный порт Mock, который является тем же самым.

Вот краткий пример в вашем случае:

from unittest.mock import MagicMock
aw = aps.Request("nv1")
aw.Clear = MagicMock()
aw2 = aps.Request("nv2", aw)
assert aw.Clear.called
13 голосов
/ 30 сентября 2010

Я не знаю ничего встроенного. Это довольно просто реализовать:

class assertMethodIsCalled(object):
    def __init__(self, obj, method):
        self.obj = obj
        self.method = method

    def called(self, *args, **kwargs):
        self.method_called = True
        self.orig_method(*args, **kwargs)

    def __enter__(self):
        self.orig_method = getattr(self.obj, self.method)
        setattr(self.obj, self.method, self.called)
        self.method_called = False

    def __exit__(self, exc_type, exc_value, traceback):
        assert getattr(self.obj, self.method) == self.called,
            "method %s was modified during assertMethodIsCalled" % self.method

        setattr(self.obj, self.method, self.orig_method)

        # If an exception was thrown within the block, we've already failed.
        if traceback is None:
            assert self.method_called,
                "method %s of %s was not called" % (self.method, self.obj)

class test(object):
    def a(self):
        print "test"
    def b(self):
        self.a()

obj = test()
with assertMethodIsCalled(obj, "a"):
    obj.b()

Это требует, чтобы сам объект не изменял self.b, что почти всегда верно.

7 голосов
/ 30 сентября 2010

Да, я могу дать вам схему, но мой Python немного ржавый, и я слишком занят, чтобы объяснять подробно.

По сути, вам нужно установить прокси в методе, который вызоветоригинал, например:

 class fred(object):
   def blog(self):
     print "We Blog"


 class methCallLogger(object):
   def __init__(self, meth):
     self.meth = meth

   def __call__(self, code=None):
     self.meth()
     # would also log the fact that it invoked the method

 #example
 f = fred()
 f.blog = methCallLogger(f.blog)

Этот ответ StackOverflow об отзыве может помочь вам понять вышесказанное.

Более подробно:

Несмотря на то, что ответ был принят, из-за интересной беседы с Гленном и нескольких свободных минут я хотел бы расширить свой ответ:

# helper class defined elsewhere
class methCallLogger(object):
   def __init__(self, meth):
     self.meth = meth
     self.was_called = False

   def __call__(self, code=None):
     self.meth()
     self.was_called = True

#example
class fred(object):
   def blog(self):
     print "We Blog"

f = fred()
g = fred()
f.blog = methCallLogger(f.blog)
g.blog = methCallLogger(g.blog)
f.blog()
assert(f.blog.was_called)
assert(not g.blog.was_called)
4 голосов
/ 30 сентября 2010

Вы можете макетировать aw.Clear, либо вручную, либо используя среду тестирования, такую ​​как pymox .Вручную, вы бы сделали это, используя что-то вроде этого:

class MyTest(TestCase):
  def testClear():
    old_clear = aw.Clear
    clear_calls = 0
    aw.Clear = lambda: clear_calls += 1
    aps.Request('nv2', aw)
    assert clear_calls == 1
    aw.Clear = old_clear

Используя pymox, вы сделали бы это так:

class MyTest(mox.MoxTestBase):
  def testClear():
    aw = self.m.CreateMock(aps.Request)
    aw.Clear()
    self.mox.ReplayAll()
    aps.Request('nv2', aw)
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...