为模拟 asyncio.coroutine 设置 return 值

Setting return value for mocked asyncio.coroutine

我正在尝试模拟 class 我正在测试的辅助方法,但我无法设置 return 值。而是 None,这会导致进行调用的方法出错。

import unittest
import asyncio
import unittest.mock

class MyClass:

    @asyncio.coroutine
    def return_number(self):
        return 5

    @asyncio.coroutine
    def add_two(self):
        val = yield from self.return_number()
        return val + 2


class TestMyClass(unittest.TestCase):

    def setUp(self):
        self.mc = MyClass()

    def test_add(self):
        val = asyncio.get_event_loop().run_until_complete(self.mc.add_two())
        self.assertEqual(val, 7)

    def test_with_mock(self):
        """ Replace return_number with mocked method """
        mm = unittest.mock.MagicMock()
        mm.iter.return_value = 2
        self.mc.return_number = unittest.mock.Mock(return_value=mm)

        val = asyncio.get_event_loop().run_until_complete(self.mc.add_two())
        self.assertEqual(val, 4)


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

在这个例子中是否有一些方法可以模拟 return_number 使其 return 成为一个特定的值?

我找到了一个解决方案,但我更喜欢类似于上面 test_with_mock 的解决方案,我可以在其中手动设置 return 值,而不是提供函数。

def test_with_patch(self):
    """ This one seems to work """
    @asyncio.coroutine
    def fake_ret_num(self):
        return 2

    with unittest.mock.patch.object(mc.MyClass, 'return_number', 
                                    fake_ret_num) as mock_meth:

        val = asyncio.get_event_loop().run_until_complete(self.mc.add_two())
        self.assertEqual(val, 4)

您也可以做类似的事情:

def test_with_mock(self):
    """ Replace return_number with mocked method """
    with mock.patch.object(self.mc, "return_number", return_value=2):
        self.mc.return_number = asyncio.coroutine(self.mc.return_number)
        val = asyncio.get_event_loop().run_until_complete(self.mc.add_two())
        self.assertEqual(val, 4)

您还可以受益于 aiounittest 并使用其 futurized

要测试的代码:

# dummy_math.py

from asyncio import sleep

async def add(x, y):
    await sleep(666)
    return x + y

并且测试使用 AsyncTestCase 并作为模拟进行未来化 return_value:

from aiounittest import futurized, AsyncTestCase
from unittest.mock import Mock, patch

import dummy_math

class MyTest(AsyncTestCase):

    def tearDown(self):
        super().tearDown()
        patch.stopall()

    async def test_add(self):
        mock_sleep = Mock(return_value=futurized('whatever'))
        patch('dummy_math.sleep', mock_sleep).start()
        ret = await dummy_math.add(5, 6)
        self.assertEqual(ret, 11)
        mock_sleep.assert_called_once_with(666)

    async def test_fail(self):
        mock_sleep = Mock(return_value=futurized(Exception('whatever')))
        patch('dummy_math.sleep', mock_sleep).start()
        with self.assertRaises(Exception) as e:
            await dummy_math.add(5, 6)
        mock_sleep.assert_called_once_with(666)