Commit 2087023f authored by Tony Flury's avatar Tony Flury Committed by Berker Peksag

bpo-32933: Implement __iter__ method on mock_open() (GH-5974)

parent c7042224
...@@ -2095,6 +2095,10 @@ mock_open ...@@ -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=''): ...@@ -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=''): ...@@ -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): ...@@ -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): ...@@ -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): ...@@ -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): ...@@ -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):
......
:func:`unittest.mock.mock_open` now supports iteration over the file
contents. Patch by Tony Flury.
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment