diff options
author | Ned Batchelder <ned@nedbatchelder.com> | 2017-11-04 18:42:42 -0400 |
---|---|---|
committer | Ned Batchelder <ned@nedbatchelder.com> | 2017-11-04 18:42:42 -0400 |
commit | 781fe6a219e1b643c24a8612fea4c8f4fb458267 (patch) | |
tree | 8a657fcf7bacf1d10c17b4c58e3432d0e4c415da /coverage/collector.py | |
parent | fd6c77243dfa7f6ac1c9d5898195bde445d05eac (diff) | |
download | python-coveragepy-git-781fe6a219e1b643c24a8612fea4c8f4fb458267.tar.gz |
Another approach to solving the 'dictionary changed size during iteration' problem
Diffstat (limited to 'coverage/collector.py')
-rw-r--r-- | coverage/collector.py | 20 |
1 files changed, 16 insertions, 4 deletions
diff --git a/coverage/collector.py b/coverage/collector.py index 70111d39..658aed62 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)) |