使用mock_open,我可以使用with [...] as
构造从写操作捕获数据。然而,测试我所拥有的是正确的有点棘手。例如,我可以这样做:
>>> from mock import mock_open
>>> m = mock_open()
>>> with patch('__main__.open', m, create=True):
... with open('foo', 'w') as h:
... h.write('some stuff')
...
>>> m.mock_calls
[call('foo', 'w'),
call().__enter__(),
call().write('some stuff'),
call().__exit__(None, None, None)]
>>> m.assert_called_once_with('foo', 'w')
>>> handle = m()
>>> handle.write.assert_called_once_with('some stuff')
但我想把我认为应该写的和以前写的做个比较。实际上是这样的:
>>> expected = 'some stuff'
>>> assert(expected == m.all_that_was_written)
我面临的问题是不同版本的json(2.0.9与1.9)的打印方式似乎不同。不,我不能只更新到最新的json。
我得到的实际错误是:
E AssertionError: [call('Tool_000.json', 'w'),
call().__enter__(),
call().write('['),
call().write('\n '),
call().write('"1.0.0"'),
call().write(', \n '),
call().write('"2014-02-27 08:58:02"'),
call().write(', \n '),
call().write('"ook"'),
call().write('\n'),
call().write(']'),
call().__exit__(None, None, None)]
!=
[call('Tool_000.json', 'w'),
call().__enter__(),
call().write('[\n "1.0.0"'),
call().write(', \n "2014-02-27 08:58:02"'),
call().write(', \n "ook"'),
call().write('\n'),
call().write(']'),
call().__exit__(None, None, None)]
实际上,调用是不同的,但最终结果是相同的。
我正在测试的代码相当简单:
with open(get_new_file_name(), 'w') as fp:
json.dump(lst, fp)
因此,创建另一个传递文件指针的方法似乎有些过头了。
最佳答案
您可以修补open()
以返回StringIO
对象,然后检查内容。
with mock.patch('module_under_test.open', create=True) as mock_open:
stream = io.StringIO()
# patching to make getvalue() work after close() or __exit__()
stream.close = mock.Mock(return_value=None)
mock_open.return_value = stream
module_under_test.do_something() # this calls open()
contents = stream.getvalue()
assert(contents == expected)
编辑:为
stream.close
添加了补丁,以避免stream.getvalue()
出现异常。