summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorThomas Kluyver <takowl@gmail.com>2017-05-23 03:27:52 (GMT)
committerNick Coghlan <ncoghlan@gmail.com>2017-05-23 03:27:52 (GMT)
commitf9169ce6b48c7cc7cc62d9eb5e4ee1ac7066d14b (patch)
tree956e51ea17f03913fed3f61be6bb0cf9bc96e9c6
parente377416c10eb0bf055b0728cdcdc4488fdfd3b5f (diff)
downloadcpython-f9169ce6b48c7cc7cc62d9eb5e4ee1ac7066d14b.zip
cpython-f9169ce6b48c7cc7cc62d9eb5e4ee1ac7066d14b.tar.gz
cpython-f9169ce6b48c7cc7cc62d9eb5e4ee1ac7066d14b.tar.bz2
bpo-25532: Protect against infinite loops in inspect.unwrap() (#1717)
Some objects (like test mocks) auto-generate new objects on attribute access, which can lead to an infinite loop in inspect.unwrap(). Ensuring references are retained to otherwise temporary objects and capping the size of the memo dict turns this case into a conventional exception instead.
-rw-r--r--Lib/inspect.py9
-rw-r--r--Lib/test/test_inspect.py18
-rw-r--r--Misc/NEWS4
3 files changed, 28 insertions, 3 deletions
diff --git a/Lib/inspect.py b/Lib/inspect.py
index 9c072eb..9a843d6 100644
--- a/Lib/inspect.py
+++ b/Lib/inspect.py
@@ -505,13 +505,16 @@ def unwrap(func, *, stop=None):
def _is_wrapper(f):
return hasattr(f, '__wrapped__') and not stop(f)
f = func # remember the original func for error reporting
- memo = {id(f)} # Memoise by id to tolerate non-hashable objects
+ # Memoise by id to tolerate non-hashable objects, but store objects to
+ # ensure they aren't destroyed, which would allow their IDs to be reused.
+ memo = {id(f): f}
+ recursion_limit = sys.getrecursionlimit()
while _is_wrapper(func):
func = func.__wrapped__
id_func = id(func)
- if id_func in memo:
+ if (id_func in memo) or (len(memo) >= recursion_limit):
raise ValueError('wrapper loop when unwrapping {!r}'.format(f))
- memo.add(id_func)
+ memo[id_func] = func
return func
# -------------------------------------------------- source code extraction
diff --git a/Lib/test/test_inspect.py b/Lib/test/test_inspect.py
index c55efd6..350d5db 100644
--- a/Lib/test/test_inspect.py
+++ b/Lib/test/test_inspect.py
@@ -3554,6 +3554,19 @@ class TestSignatureDefinitions(unittest.TestCase):
self.assertIsNone(obj.__text_signature__)
+class NTimesUnwrappable:
+ def __init__(self, n):
+ self.n = n
+ self._next = None
+
+ @property
+ def __wrapped__(self):
+ if self.n <= 0:
+ raise Exception("Unwrapped too many times")
+ if self._next is None:
+ self._next = NTimesUnwrappable(self.n - 1)
+ return self._next
+
class TestUnwrap(unittest.TestCase):
def test_unwrap_one(self):
@@ -3609,6 +3622,11 @@ class TestUnwrap(unittest.TestCase):
__wrapped__ = func
self.assertIsNone(inspect.unwrap(C()))
+ def test_recursion_limit(self):
+ obj = NTimesUnwrappable(sys.getrecursionlimit() + 1)
+ with self.assertRaisesRegex(ValueError, 'wrapper loop'):
+ inspect.unwrap(obj)
+
class TestMain(unittest.TestCase):
def test_only_source(self):
module = importlib.import_module('unittest')
diff --git a/Misc/NEWS b/Misc/NEWS
index 5b41dcf..0e32d48 100644
--- a/Misc/NEWS
+++ b/Misc/NEWS
@@ -995,6 +995,10 @@ Library
- Issue #29581: ABCMeta.__new__ now accepts ``**kwargs``, allowing abstract base
classes to use keyword parameters in __init_subclass__. Patch by Nate Soares.
+- Issue #25532: inspect.unwrap() will now only try to unwrap an object
+ sys.getrecursionlimit() times, to protect against objects which create a new
+ object on every attribute access.
+
Windows
-------