diff options
author | Meador Inge <meadori@gmail.com> | 2015-07-23 22:52:49 -0500 |
---|---|---|
committer | Meador Inge <meadori@gmail.com> | 2015-07-23 22:52:49 -0500 |
commit | f98c35a8160e7e373da0de4f348bbeaa3b718041 (patch) | |
tree | a67a34548e2ca3d18bea7908fd94b7a0705a4de7 /Lib/inspect.py | |
parent | Add versionchanged information for mock_open. (diff) | |
parent | Issue #24485: Function source inspection fails on closures. (diff) | |
download | cpython-f98c35a8160e7e373da0de4f348bbeaa3b718041.tar.gz cpython-f98c35a8160e7e373da0de4f348bbeaa3b718041.tar.bz2 cpython-f98c35a8160e7e373da0de4f348bbeaa3b718041.zip |
Issue #24485: Function source inspection fails on closures.
The fix for Issue #21217 introduced a regression that caused
`inspect.getsource` to return incorrect results on nested
functions. The root cause of the regression was due to
switching the implementation to analyze the underlying
bytecode instead of the source code.
This commit switches things back to analyzing the source code
in a more complete way. The original bug and the regression
are both fixed by the new source code analysis.
Diffstat (limited to 'Lib/inspect.py')
-rw-r--r-- | Lib/inspect.py | 26 |
1 files changed, 22 insertions, 4 deletions
diff --git a/Lib/inspect.py b/Lib/inspect.py index 305aafe6ed7..1e94a9c539b 100644 --- a/Lib/inspect.py +++ b/Lib/inspect.py @@ -842,21 +842,37 @@ class BlockFinder: self.islambda = False self.started = False self.passline = False + self.indecorator = False + self.decoratorhasargs = False self.last = 1 def tokeneater(self, type, token, srowcol, erowcol, line): - if not self.started: + if not self.started and not self.indecorator: + # skip any decorators + if token == "@": + self.indecorator = True # look for the first "def", "class" or "lambda" - if token in ("def", "class", "lambda"): + elif token in ("def", "class", "lambda"): if token == "lambda": self.islambda = True self.started = True self.passline = True # skip to the end of the line + elif token == "(": + if self.indecorator: + self.decoratorhasargs = True + elif token == ")": + if self.indecorator: + self.indecorator = False + self.decoratorhasargs = False elif type == tokenize.NEWLINE: self.passline = False # stop skipping when a NEWLINE is seen self.last = srowcol[0] if self.islambda: # lambdas always end at the first NEWLINE raise EndOfBlock + # hitting a NEWLINE when in a decorator without args + # ends the decorator + if self.indecorator and not self.decoratorhasargs: + self.indecorator = False elif self.passline: pass elif type == tokenize.INDENT: @@ -896,8 +912,10 @@ def getsourcelines(object): object = unwrap(object) lines, lnum = findsource(object) - if ismodule(object): return lines, 0 - else: return getblock(lines[lnum:]), lnum + 1 + if ismodule(object): + return lines, 0 + else: + return getblock(lines[lnum:]), lnum + 1 def getsource(object): """Return the text of the source code for an object. |