mirror of
				https://github.com/python/cpython.git
				synced 2025-10-26 16:27:06 +00:00 
			
		
		
		
	 394119afc6
			
		
	
	
		394119afc6
		
	
	
	
	
		
			
			I've reported the issue on https://bugs.python.org/issue37008 and now I'm trying to bring a solution to this minor issue. I think it could be trivially backported to 3.7 branch. https://bugs.python.org/issue37008
		
			
				
	
	
		
			347 lines
		
	
	
	
		
			12 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			347 lines
		
	
	
	
		
			12 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| import unittest
 | |
| from warnings import catch_warnings
 | |
| 
 | |
| from unittest.test.testmock.support import is_instance
 | |
| from unittest.mock import MagicMock, Mock, patch, sentinel, mock_open, call
 | |
| 
 | |
| 
 | |
| 
 | |
| something  = sentinel.Something
 | |
| something_else  = sentinel.SomethingElse
 | |
| 
 | |
| 
 | |
| class SampleException(Exception): pass
 | |
| 
 | |
| 
 | |
| class WithTest(unittest.TestCase):
 | |
| 
 | |
|     def test_with_statement(self):
 | |
|         with patch('%s.something' % __name__, sentinel.Something2):
 | |
|             self.assertEqual(something, sentinel.Something2, "unpatched")
 | |
|         self.assertEqual(something, sentinel.Something)
 | |
| 
 | |
| 
 | |
|     def test_with_statement_exception(self):
 | |
|         with self.assertRaises(SampleException):
 | |
|             with patch('%s.something' % __name__, sentinel.Something2):
 | |
|                 self.assertEqual(something, sentinel.Something2, "unpatched")
 | |
|                 raise SampleException()
 | |
|         self.assertEqual(something, sentinel.Something)
 | |
| 
 | |
| 
 | |
|     def test_with_statement_as(self):
 | |
|         with patch('%s.something' % __name__) as mock_something:
 | |
|             self.assertEqual(something, mock_something, "unpatched")
 | |
|             self.assertTrue(is_instance(mock_something, MagicMock),
 | |
|                             "patching wrong type")
 | |
|         self.assertEqual(something, sentinel.Something)
 | |
| 
 | |
| 
 | |
|     def test_patch_object_with_statement(self):
 | |
|         class Foo(object):
 | |
|             something = 'foo'
 | |
|         original = Foo.something
 | |
|         with patch.object(Foo, 'something'):
 | |
|             self.assertNotEqual(Foo.something, original, "unpatched")
 | |
|         self.assertEqual(Foo.something, original)
 | |
| 
 | |
| 
 | |
|     def test_with_statement_nested(self):
 | |
|         with catch_warnings(record=True):
 | |
|             with patch('%s.something' % __name__) as mock_something, patch('%s.something_else' % __name__) as mock_something_else:
 | |
|                 self.assertEqual(something, mock_something, "unpatched")
 | |
|                 self.assertEqual(something_else, mock_something_else,
 | |
|                                  "unpatched")
 | |
| 
 | |
|         self.assertEqual(something, sentinel.Something)
 | |
|         self.assertEqual(something_else, sentinel.SomethingElse)
 | |
| 
 | |
| 
 | |
|     def test_with_statement_specified(self):
 | |
|         with patch('%s.something' % __name__, sentinel.Patched) as mock_something:
 | |
|             self.assertEqual(something, mock_something, "unpatched")
 | |
|             self.assertEqual(mock_something, sentinel.Patched, "wrong patch")
 | |
|         self.assertEqual(something, sentinel.Something)
 | |
| 
 | |
| 
 | |
|     def testContextManagerMocking(self):
 | |
|         mock = Mock()
 | |
|         mock.__enter__ = Mock()
 | |
|         mock.__exit__ = Mock()
 | |
|         mock.__exit__.return_value = False
 | |
| 
 | |
|         with mock as m:
 | |
|             self.assertEqual(m, mock.__enter__.return_value)
 | |
|         mock.__enter__.assert_called_with()
 | |
|         mock.__exit__.assert_called_with(None, None, None)
 | |
| 
 | |
| 
 | |
|     def test_context_manager_with_magic_mock(self):
 | |
|         mock = MagicMock()
 | |
| 
 | |
|         with self.assertRaises(TypeError):
 | |
|             with mock:
 | |
|                 'foo' + 3
 | |
|         mock.__enter__.assert_called_with()
 | |
|         self.assertTrue(mock.__exit__.called)
 | |
| 
 | |
| 
 | |
|     def test_with_statement_same_attribute(self):
 | |
|         with patch('%s.something' % __name__, sentinel.Patched) as mock_something:
 | |
|             self.assertEqual(something, mock_something, "unpatched")
 | |
| 
 | |
|             with patch('%s.something' % __name__) as mock_again:
 | |
|                 self.assertEqual(something, mock_again, "unpatched")
 | |
| 
 | |
|             self.assertEqual(something, mock_something,
 | |
|                              "restored with wrong instance")
 | |
| 
 | |
|         self.assertEqual(something, sentinel.Something, "not restored")
 | |
| 
 | |
| 
 | |
|     def test_with_statement_imbricated(self):
 | |
|         with patch('%s.something' % __name__) as mock_something:
 | |
|             self.assertEqual(something, mock_something, "unpatched")
 | |
| 
 | |
|             with patch('%s.something_else' % __name__) as mock_something_else:
 | |
|                 self.assertEqual(something_else, mock_something_else,
 | |
|                                  "unpatched")
 | |
| 
 | |
|         self.assertEqual(something, sentinel.Something)
 | |
|         self.assertEqual(something_else, sentinel.SomethingElse)
 | |
| 
 | |
| 
 | |
|     def test_dict_context_manager(self):
 | |
|         foo = {}
 | |
|         with patch.dict(foo, {'a': 'b'}):
 | |
|             self.assertEqual(foo, {'a': 'b'})
 | |
|         self.assertEqual(foo, {})
 | |
| 
 | |
|         with self.assertRaises(NameError):
 | |
|             with patch.dict(foo, {'a': 'b'}):
 | |
|                 self.assertEqual(foo, {'a': 'b'})
 | |
|                 raise NameError('Konrad')
 | |
| 
 | |
|         self.assertEqual(foo, {})
 | |
| 
 | |
|     def test_double_patch_instance_method(self):
 | |
|         class C:
 | |
|             def f(self): pass
 | |
| 
 | |
|         c = C()
 | |
| 
 | |
|         with patch.object(c, 'f', autospec=True) as patch1:
 | |
|             with patch.object(c, 'f', autospec=True) as patch2:
 | |
|                 c.f()
 | |
|             self.assertEqual(patch2.call_count, 1)
 | |
|             self.assertEqual(patch1.call_count, 0)
 | |
|             c.f()
 | |
|         self.assertEqual(patch1.call_count, 1)
 | |
| 
 | |
| 
 | |
| class TestMockOpen(unittest.TestCase):
 | |
| 
 | |
|     def test_mock_open(self):
 | |
|         mock = mock_open()
 | |
|         with patch('%s.open' % __name__, mock, create=True) as patched:
 | |
|             self.assertIs(patched, mock)
 | |
|             open('foo')
 | |
| 
 | |
|         mock.assert_called_once_with('foo')
 | |
| 
 | |
| 
 | |
|     def test_mock_open_context_manager(self):
 | |
|         mock = mock_open()
 | |
|         handle = mock.return_value
 | |
|         with patch('%s.open' % __name__, mock, create=True):
 | |
|             with open('foo') as f:
 | |
|                 f.read()
 | |
| 
 | |
|         expected_calls = [call('foo'), call().__enter__(), call().read(),
 | |
|                           call().__exit__(None, None, None)]
 | |
|         self.assertEqual(mock.mock_calls, expected_calls)
 | |
|         self.assertIs(f, handle)
 | |
| 
 | |
|     def test_mock_open_context_manager_multiple_times(self):
 | |
|         mock = mock_open()
 | |
|         with patch('%s.open' % __name__, mock, create=True):
 | |
|             with open('foo') as f:
 | |
|                 f.read()
 | |
|             with open('bar') as f:
 | |
|                 f.read()
 | |
| 
 | |
|         expected_calls = [
 | |
|             call('foo'), call().__enter__(), call().read(),
 | |
|             call().__exit__(None, None, None),
 | |
|             call('bar'), call().__enter__(), call().read(),
 | |
|             call().__exit__(None, None, None)]
 | |
|         self.assertEqual(mock.mock_calls, expected_calls)
 | |
| 
 | |
|     def test_explicit_mock(self):
 | |
|         mock = MagicMock()
 | |
|         mock_open(mock)
 | |
| 
 | |
|         with patch('%s.open' % __name__, mock, create=True) as patched:
 | |
|             self.assertIs(patched, mock)
 | |
|             open('foo')
 | |
| 
 | |
|         mock.assert_called_once_with('foo')
 | |
| 
 | |
| 
 | |
|     def test_read_data(self):
 | |
|         mock = mock_open(read_data='foo')
 | |
|         with patch('%s.open' % __name__, mock, create=True):
 | |
|             h = open('bar')
 | |
|             result = h.read()
 | |
| 
 | |
|         self.assertEqual(result, 'foo')
 | |
| 
 | |
| 
 | |
|     def test_readline_data(self):
 | |
|         # 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')
 | |
|         with patch('%s.open' % __name__, mock, create=True):
 | |
|             h = open('bar')
 | |
|             line1 = h.readline()
 | |
|             line2 = h.readline()
 | |
|             line3 = h.readline()
 | |
|         self.assertEqual(line1, 'foo\n')
 | |
|         self.assertEqual(line2, 'bar\n')
 | |
|         self.assertEqual(line3, 'baz\n')
 | |
|         self.assertEqual(h.readline(), '')
 | |
| 
 | |
|         # Check that we properly emulate a file that doesn't end in a newline
 | |
|         mock = mock_open(read_data='foo')
 | |
|         with patch('%s.open' % __name__, mock, create=True):
 | |
|             h = open('bar')
 | |
|             result = h.readline()
 | |
|         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(), '')
 | |
|         with self.assertRaises(StopIteration):
 | |
|             next(h)
 | |
| 
 | |
|     def test_next_data(self):
 | |
|         # Check that next will correctly return the next available
 | |
|         # line and plays well with the dunder_iter part.
 | |
|         mock = mock_open(read_data='foo\nbar\nbaz\n')
 | |
|         with patch('%s.open' % __name__, mock, create=True):
 | |
|             h = open('bar')
 | |
|             line1 = next(h)
 | |
|             line2 = next(h)
 | |
|             lines = [l for l in h]
 | |
|         self.assertEqual(line1, 'foo\n')
 | |
|         self.assertEqual(line2, 'bar\n')
 | |
|         self.assertEqual(lines[0], 'baz\n')
 | |
|         self.assertEqual(h.readline(), '')
 | |
| 
 | |
|     def test_readlines_data(self):
 | |
|         # Test that emulating a file that ends in a newline character works
 | |
|         mock = mock_open(read_data='foo\nbar\nbaz\n')
 | |
|         with patch('%s.open' % __name__, mock, create=True):
 | |
|             h = open('bar')
 | |
|             result = h.readlines()
 | |
|         self.assertEqual(result, ['foo\n', 'bar\n', 'baz\n'])
 | |
| 
 | |
|         # Test that files without a final newline will also be correctly
 | |
|         # emulated
 | |
|         mock = mock_open(read_data='foo\nbar\nbaz')
 | |
|         with patch('%s.open' % __name__, mock, create=True):
 | |
|             h = open('bar')
 | |
|             result = h.readlines()
 | |
| 
 | |
|         self.assertEqual(result, ['foo\n', 'bar\n', 'baz'])
 | |
| 
 | |
| 
 | |
|     def test_read_bytes(self):
 | |
|         mock = mock_open(read_data=b'\xc6')
 | |
|         with patch('%s.open' % __name__, mock, create=True):
 | |
|             with open('abc', 'rb') as f:
 | |
|                 result = f.read()
 | |
|         self.assertEqual(result, b'\xc6')
 | |
| 
 | |
| 
 | |
|     def test_readline_bytes(self):
 | |
|         m = mock_open(read_data=b'abc\ndef\nghi\n')
 | |
|         with patch('%s.open' % __name__, m, create=True):
 | |
|             with open('abc', 'rb') as f:
 | |
|                 line1 = f.readline()
 | |
|                 line2 = f.readline()
 | |
|                 line3 = f.readline()
 | |
|         self.assertEqual(line1, b'abc\n')
 | |
|         self.assertEqual(line2, b'def\n')
 | |
|         self.assertEqual(line3, b'ghi\n')
 | |
| 
 | |
| 
 | |
|     def test_readlines_bytes(self):
 | |
|         m = mock_open(read_data=b'abc\ndef\nghi\n')
 | |
|         with patch('%s.open' % __name__, m, create=True):
 | |
|             with open('abc', 'rb') as f:
 | |
|                 result = f.readlines()
 | |
|         self.assertEqual(result, [b'abc\n', b'def\n', b'ghi\n'])
 | |
| 
 | |
| 
 | |
|     def test_mock_open_read_with_argument(self):
 | |
|         # At one point calling read with an argument was broken
 | |
|         # for mocks returned by mock_open
 | |
|         some_data = 'foo\nbar\nbaz'
 | |
|         mock = mock_open(read_data=some_data)
 | |
|         self.assertEqual(mock().read(10), some_data[:10])
 | |
|         self.assertEqual(mock().read(10), some_data[:10])
 | |
| 
 | |
|         f = mock()
 | |
|         self.assertEqual(f.read(10), some_data[:10])
 | |
|         self.assertEqual(f.read(10), some_data[10:])
 | |
| 
 | |
| 
 | |
|     def test_interleaved_reads(self):
 | |
|         # Test that calling read, readline, and readlines pulls data
 | |
|         # sequentially from the data we preload with
 | |
|         mock = mock_open(read_data='foo\nbar\nbaz\n')
 | |
|         with patch('%s.open' % __name__, mock, create=True):
 | |
|             h = open('bar')
 | |
|             line1 = h.readline()
 | |
|             rest = h.readlines()
 | |
|         self.assertEqual(line1, 'foo\n')
 | |
|         self.assertEqual(rest, ['bar\n', 'baz\n'])
 | |
| 
 | |
|         mock = mock_open(read_data='foo\nbar\nbaz\n')
 | |
|         with patch('%s.open' % __name__, mock, create=True):
 | |
|             h = open('bar')
 | |
|             line1 = h.readline()
 | |
|             rest = h.read()
 | |
|         self.assertEqual(line1, 'foo\n')
 | |
|         self.assertEqual(rest, 'bar\nbaz\n')
 | |
| 
 | |
| 
 | |
|     def test_overriding_return_values(self):
 | |
|         mock = mock_open(read_data='foo')
 | |
|         handle = mock()
 | |
| 
 | |
|         handle.read.return_value = 'bar'
 | |
|         handle.readline.return_value = 'bar'
 | |
|         handle.readlines.return_value = ['bar']
 | |
| 
 | |
|         self.assertEqual(handle.read(), 'bar')
 | |
|         self.assertEqual(handle.readline(), 'bar')
 | |
|         self.assertEqual(handle.readlines(), ['bar'])
 | |
| 
 | |
|         # call repeatedly to check that a StopIteration is not propagated
 | |
|         self.assertEqual(handle.readline(), 'bar')
 | |
|         self.assertEqual(handle.readline(), 'bar')
 | |
| 
 | |
| 
 | |
| if __name__ == '__main__':
 | |
|     unittest.main()
 |