summaryrefslogtreecommitdiffstats
path: root/Lib
diff options
context:
space:
mode:
Diffstat (limited to 'Lib')
-rw-r--r--Lib/test/test_unittest/testmock/testpatch.py52
-rw-r--r--Lib/unittest/mock.py9
2 files changed, 59 insertions, 2 deletions
diff --git a/Lib/test/test_unittest/testmock/testpatch.py b/Lib/test/test_unittest/testmock/testpatch.py
index f26e74c..037c021 100644
--- a/Lib/test/test_unittest/testmock/testpatch.py
+++ b/Lib/test/test_unittest/testmock/testpatch.py
@@ -745,6 +745,54 @@ class PatchTest(unittest.TestCase):
self.assertIsNone(patcher.stop())
+ def test_exit_idempotent(self):
+ patcher = patch(foo_name, 'bar', 3)
+ with patcher:
+ patcher.stop()
+
+
+ def test_second_start_failure(self):
+ patcher = patch(foo_name, 'bar', 3)
+ patcher.start()
+ try:
+ self.assertRaises(RuntimeError, patcher.start)
+ finally:
+ patcher.stop()
+
+
+ def test_second_enter_failure(self):
+ patcher = patch(foo_name, 'bar', 3)
+ with patcher:
+ self.assertRaises(RuntimeError, patcher.start)
+
+
+ def test_second_start_after_stop(self):
+ patcher = patch(foo_name, 'bar', 3)
+ patcher.start()
+ patcher.stop()
+ patcher.start()
+ patcher.stop()
+
+
+ def test_property_setters(self):
+ mock_object = Mock()
+ mock_bar = mock_object.bar
+ patcher = patch.object(mock_object, 'bar', 'x')
+ with patcher:
+ self.assertEqual(patcher.is_local, False)
+ self.assertIs(patcher.target, mock_object)
+ self.assertEqual(patcher.temp_original, mock_bar)
+ patcher.is_local = True
+ patcher.target = mock_bar
+ patcher.temp_original = mock_object
+ self.assertEqual(patcher.is_local, True)
+ self.assertIs(patcher.target, mock_bar)
+ self.assertEqual(patcher.temp_original, mock_object)
+ # if changes are left intact, they may lead to disruption as shown below (it might be what someone needs though)
+ self.assertEqual(mock_bar.bar, mock_object)
+ self.assertEqual(mock_object.bar, 'x')
+
+
def test_patchobject_start_stop(self):
original = something
patcher = patch.object(PTModule, 'something', 'foo')
@@ -1098,7 +1146,7 @@ class PatchTest(unittest.TestCase):
self.assertIsNot(m1, m2)
for mock in m1, m2:
- self.assertNotCallable(m1)
+ self.assertNotCallable(mock)
def test_new_callable_patch_object(self):
@@ -1111,7 +1159,7 @@ class PatchTest(unittest.TestCase):
self.assertIsNot(m1, m2)
for mock in m1, m2:
- self.assertNotCallable(m1)
+ self.assertNotCallable(mock)
def test_new_callable_keyword_arguments(self):
diff --git a/Lib/unittest/mock.py b/Lib/unittest/mock.py
index 21ca061..55cb4b1 100644
--- a/Lib/unittest/mock.py
+++ b/Lib/unittest/mock.py
@@ -1360,6 +1360,7 @@ class _patch(object):
self.autospec = autospec
self.kwargs = kwargs
self.additional_patchers = []
+ self.is_started = False
def copy(self):
@@ -1472,6 +1473,9 @@ class _patch(object):
def __enter__(self):
"""Perform the patch."""
+ if self.is_started:
+ raise RuntimeError("Patch is already started")
+
new, spec, spec_set = self.new, self.spec, self.spec_set
autospec, kwargs = self.autospec, self.kwargs
new_callable = self.new_callable
@@ -1603,6 +1607,7 @@ class _patch(object):
self.temp_original = original
self.is_local = local
self._exit_stack = contextlib.ExitStack()
+ self.is_started = True
try:
setattr(self.target, self.attribute, new_attr)
if self.attribute_name is not None:
@@ -1622,6 +1627,9 @@ class _patch(object):
def __exit__(self, *exc_info):
"""Undo the patch."""
+ if not self.is_started:
+ return
+
if self.is_local and self.temp_original is not DEFAULT:
setattr(self.target, self.attribute, self.temp_original)
else:
@@ -1638,6 +1646,7 @@ class _patch(object):
del self.target
exit_stack = self._exit_stack
del self._exit_stack
+ self.is_started = False
return exit_stack.__exit__(*exc_info)