为模拟 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)
我正在尝试模拟 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)