mirror of
https://github.com/python/cpython.git
synced 2025-10-13 10:23:28 +00:00
bpo-32933: Implement __iter__ method on mock_open() (GH-5974)
This commit is contained in:
parent
c7042224b8
commit
2087023fde
5 changed files with 37 additions and 3 deletions
|
@ -2095,6 +2095,10 @@ mock_open
|
||||||
.. versionchanged:: 3.5
|
.. versionchanged:: 3.5
|
||||||
*read_data* is now reset on each call to the *mock*.
|
*read_data* is now reset on each call to the *mock*.
|
||||||
|
|
||||||
|
.. versionchanged:: 3.8
|
||||||
|
Added :meth:`__iter__` to implementation so that iteration (such as in for
|
||||||
|
loops) correctly consumes *read_data*.
|
||||||
|
|
||||||
Using :func:`open` as a context manager is a great way to ensure your file handles
|
Using :func:`open` as a context manager is a great way to ensure your file handles
|
||||||
are closed properly and is becoming common::
|
are closed properly and is becoming common::
|
||||||
|
|
||||||
|
|
|
@ -2358,14 +2358,16 @@ def mock_open(mock=None, read_data=''):
|
||||||
return type(read_data)().join(_state[0])
|
return type(read_data)().join(_state[0])
|
||||||
|
|
||||||
def _readline_side_effect():
|
def _readline_side_effect():
|
||||||
|
yield from _iter_side_effect()
|
||||||
|
while True:
|
||||||
|
yield type(read_data)()
|
||||||
|
|
||||||
|
def _iter_side_effect():
|
||||||
if handle.readline.return_value is not None:
|
if handle.readline.return_value is not None:
|
||||||
while True:
|
while True:
|
||||||
yield handle.readline.return_value
|
yield handle.readline.return_value
|
||||||
for line in _state[0]:
|
for line in _state[0]:
|
||||||
yield line
|
yield line
|
||||||
while True:
|
|
||||||
yield type(read_data)()
|
|
||||||
|
|
||||||
|
|
||||||
global file_spec
|
global file_spec
|
||||||
if file_spec is None:
|
if file_spec is None:
|
||||||
|
@ -2389,6 +2391,7 @@ def mock_open(mock=None, read_data=''):
|
||||||
_state[1] = _readline_side_effect()
|
_state[1] = _readline_side_effect()
|
||||||
handle.readline.side_effect = _state[1]
|
handle.readline.side_effect = _state[1]
|
||||||
handle.readlines.side_effect = _readlines_side_effect
|
handle.readlines.side_effect = _readlines_side_effect
|
||||||
|
handle.__iter__.side_effect = _iter_side_effect
|
||||||
|
|
||||||
def reset_data(*args, **kwargs):
|
def reset_data(*args, **kwargs):
|
||||||
_state[0] = _iterate_read_data(read_data)
|
_state[0] = _iterate_read_data(read_data)
|
||||||
|
|
|
@ -1450,6 +1450,16 @@ class MockTest(unittest.TestCase):
|
||||||
f2_data = f2.read()
|
f2_data = f2.read()
|
||||||
self.assertEqual(f1_data, f2_data)
|
self.assertEqual(f1_data, f2_data)
|
||||||
|
|
||||||
|
def test_mock_open_dunder_iter_issue(self):
|
||||||
|
# Test dunder_iter method generates the expected result and
|
||||||
|
# consumes the iterator.
|
||||||
|
mocked_open = mock.mock_open(read_data='Remarkable\nNorwegian Blue')
|
||||||
|
f1 = mocked_open('a-name')
|
||||||
|
lines = [line for line in f1]
|
||||||
|
self.assertEqual(lines[0], 'Remarkable\n')
|
||||||
|
self.assertEqual(lines[1], 'Norwegian Blue')
|
||||||
|
self.assertEqual(list(f1), [])
|
||||||
|
|
||||||
def test_mock_open_write(self):
|
def test_mock_open_write(self):
|
||||||
# Test exception in file writing write()
|
# Test exception in file writing write()
|
||||||
mock_namedtemp = mock.mock_open(mock.MagicMock(name='JLV'))
|
mock_namedtemp = mock.mock_open(mock.MagicMock(name='JLV'))
|
||||||
|
|
|
@ -188,6 +188,7 @@ class TestMockOpen(unittest.TestCase):
|
||||||
|
|
||||||
def test_readline_data(self):
|
def test_readline_data(self):
|
||||||
# Check that readline will return all the lines from the fake file
|
# Check that readline will return all the lines from the fake file
|
||||||
|
# And that once fully consumed, readline will return an empty string.
|
||||||
mock = mock_open(read_data='foo\nbar\nbaz\n')
|
mock = mock_open(read_data='foo\nbar\nbaz\n')
|
||||||
with patch('%s.open' % __name__, mock, create=True):
|
with patch('%s.open' % __name__, mock, create=True):
|
||||||
h = open('bar')
|
h = open('bar')
|
||||||
|
@ -197,6 +198,7 @@ class TestMockOpen(unittest.TestCase):
|
||||||
self.assertEqual(line1, 'foo\n')
|
self.assertEqual(line1, 'foo\n')
|
||||||
self.assertEqual(line2, 'bar\n')
|
self.assertEqual(line2, 'bar\n')
|
||||||
self.assertEqual(line3, 'baz\n')
|
self.assertEqual(line3, 'baz\n')
|
||||||
|
self.assertEqual(h.readline(), '')
|
||||||
|
|
||||||
# Check that we properly emulate a file that doesn't end in a newline
|
# Check that we properly emulate a file that doesn't end in a newline
|
||||||
mock = mock_open(read_data='foo')
|
mock = mock_open(read_data='foo')
|
||||||
|
@ -204,6 +206,19 @@ class TestMockOpen(unittest.TestCase):
|
||||||
h = open('bar')
|
h = open('bar')
|
||||||
result = h.readline()
|
result = h.readline()
|
||||||
self.assertEqual(result, 'foo')
|
self.assertEqual(result, 'foo')
|
||||||
|
self.assertEqual(h.readline(), '')
|
||||||
|
|
||||||
|
|
||||||
|
def test_dunder_iter_data(self):
|
||||||
|
# Check that dunder_iter will return all the lines from the fake file.
|
||||||
|
mock = mock_open(read_data='foo\nbar\nbaz\n')
|
||||||
|
with patch('%s.open' % __name__, mock, create=True):
|
||||||
|
h = open('bar')
|
||||||
|
lines = [l for l in h]
|
||||||
|
self.assertEqual(lines[0], 'foo\n')
|
||||||
|
self.assertEqual(lines[1], 'bar\n')
|
||||||
|
self.assertEqual(lines[2], 'baz\n')
|
||||||
|
self.assertEqual(h.readline(), '')
|
||||||
|
|
||||||
|
|
||||||
def test_readlines_data(self):
|
def test_readlines_data(self):
|
||||||
|
|
|
@ -0,0 +1,2 @@
|
||||||
|
:func:`unittest.mock.mock_open` now supports iteration over the file
|
||||||
|
contents. Patch by Tony Flury.
|
Loading…
Add table
Add a link
Reference in a new issue