summaryrefslogtreecommitdiff
path: root/coverage/collector.py
diff options
context:
space:
mode:
authorNed Batchelder <ned@nedbatchelder.com>2017-11-04 18:42:42 -0400
committerNed Batchelder <ned@nedbatchelder.com>2017-11-04 18:42:42 -0400
commit6f2b0f613c3770b2fe9b702ff691dd5e46c1f44d (patch)
tree2aac9f9405a98f01ba16be63d6138827e1dddb4e /coverage/collector.py
parent1523ffa9e5f26a782981409ecb784e24e074f250 (diff)
downloadpython-coveragepy-6f2b0f613c3770b2fe9b702ff691dd5e46c1f44d.tar.gz
Another approach to solving the 'dictionary changed size during iteration' problem
Diffstat (limited to 'coverage/collector.py')
-rw-r--r--coverage/collector.py20
1 files changed, 16 insertions, 4 deletions
diff --git a/coverage/collector.py b/coverage/collector.py
index 70111d3..658aed6 100644
--- a/coverage/collector.py
+++ b/coverage/collector.py
@@ -7,7 +7,7 @@ import os
import sys
from coverage import env
-from coverage.backward import iitems
+from coverage.backward import litems, range # pylint: disable=redefined-builtin
from coverage.debug import short_stack
from coverage.files import abs_file
from coverage.misc import CoverageException, isolate_module
@@ -382,10 +382,22 @@ class Collector(object):
def abs_file_dict(d):
"""Return a dict like d, but with keys modified by `abs_file`."""
- # The call to list() ensures that the GIL protects the dictionary
+ # The call to litems() ensures that the GIL protects the dictionary
# iterator against concurrent modifications by tracers running
- # in other threads.
- return dict((abs_file(k), v) for k, v in list(iitems(d)))
+ # in other threads. We try three times in case of concurrent
+ # access, hoping to get a clean copy.
+ runtime_err = None
+ for _ in range(3):
+ try:
+ items = litems(d)
+ except RuntimeError as ex:
+ runtime_err = ex
+ else:
+ break
+ else:
+ raise runtime_err # pylint: disable=raising-bad-type
+
+ return dict((abs_file(k), v) for k, v in items)
if self.branch:
covdata.add_arcs(abs_file_dict(self.data))