summaryrefslogtreecommitdiff
path: root/coverage/parser.py
diff options
context:
space:
mode:
authorNed Batchelder <ned@nedbatchelder.com>2016-01-09 17:34:32 -0500
committerNed Batchelder <ned@nedbatchelder.com>2016-01-09 17:34:32 -0500
commit12f4082afee3900e0518088997b9bff5f93db53c (patch)
tree10d728aee7f42637ce5ff482e2a4262f1aed4eea /coverage/parser.py
parent772b6c10ff3346350ba97c1e82231feb48edb23c (diff)
downloadpython-coveragepy-12f4082afee3900e0518088997b9bff5f93db53c.tar.gz
Properly skip lines that are optimized away
Diffstat (limited to 'coverage/parser.py')
-rw-r--r--coverage/parser.py21
1 files changed, 12 insertions, 9 deletions
diff --git a/coverage/parser.py b/coverage/parser.py
index 4f15743..501b76c 100644
--- a/coverage/parser.py
+++ b/coverage/parser.py
@@ -75,7 +75,8 @@ class PythonParser(object):
# Internal detail, used by lab/parser.py.
self.show_tokens = False
- # A dict mapping line numbers to (lo,hi) for multi-line statements.
+ # A dict mapping line numbers to lexical statement starts for
+ # multi-line statements.
self._multiline = {}
# Lazily-created ByteParser and arc data.
@@ -200,11 +201,7 @@ class PythonParser(object):
def first_line(self, line):
"""Return the first line number of the statement including `line`."""
- first_line = self._multiline.get(line)
- if first_line:
- return first_line
- else:
- return line
+ return self._multiline.get(line, line)
def first_lines(self, lines):
"""Map the line numbers in `lines` to the correct first line of the
@@ -257,7 +254,7 @@ class PythonParser(object):
"""
if self._all_arcs is None:
- aaa = AstArcAnalyzer(self.text, self.raw_statements)
+ aaa = AstArcAnalyzer(self.text, self.raw_statements, self._multiline)
arcs = aaa.collect_arcs()
self._all_arcs = set()
@@ -328,12 +325,15 @@ class AstArcAnalyzer(object):
"""Analyze source text with an AST to find executable code paths."""
@contract(text='unicode', statements=set)
- def __init__(self, text, statements):
+ def __init__(self, text, statements, multiline):
self.root_node = ast.parse(neuter_encoding_declaration(text))
- self.statements = statements
+ # TODO: I think this is happening in too many places.
+ self.statements = set(multiline.get(l, l) for l in statements)
+ self.multiline = multiline
if int(os.environ.get("COVERAGE_ASTDUMP", 0)): # pragma: debugging
# Dump the AST so that failing tests have helpful output.
+ print(self.statements)
ast_dump(self.root_node)
self.arcs = None
@@ -419,6 +419,9 @@ class AstArcAnalyzer(object):
prev_lines = set([from_line])
for body_node in body:
lineno = self.line_for_node(body_node)
+ first_line = self.multiline.get(lineno, lineno)
+ if first_line not in self.statements:
+ continue
for prev_lineno in prev_lines:
self.arcs.add((prev_lineno, lineno))
prev_lines = self.add_arcs(body_node)