diff options
| author | Antoine Pitrou <solipsis@pitrou.net> | 2014-10-05 20:02:28 +0200 | 
|---|---|---|
| committer | Antoine Pitrou <solipsis@pitrou.net> | 2014-10-05 20:02:28 +0200 | 
| commit | 1bf974dc6e15799856b1e5d6316238ecedcd798f (patch) | |
| tree | 705dd21d7ea676aaa786ef40eecd2350f8d5e13a /Lib/weakref.py | |
| parent | bed04a77ee4dde8e1ab7b00557519f01cac734aa (diff) | |
| download | cpython-git-1bf974dc6e15799856b1e5d6316238ecedcd798f.tar.gz | |
Closes #21173: Fix len() on a WeakKeyDictionary when .clear() was called with an iterator alive.
Diffstat (limited to 'Lib/weakref.py')
| -rw-r--r-- | Lib/weakref.py | 13 | 
1 files changed, 13 insertions, 0 deletions
| diff --git a/Lib/weakref.py b/Lib/weakref.py index f6a40ca4bf..12bf9754c5 100644 --- a/Lib/weakref.py +++ b/Lib/weakref.py @@ -322,6 +322,7 @@ class WeakKeyDictionary(collections.MutableMapping):          # A list of dead weakrefs (keys to be removed)          self._pending_removals = []          self._iterating = set() +        self._dirty_len = False          if dict is not None:              self.update(dict) @@ -338,13 +339,23 @@ class WeakKeyDictionary(collections.MutableMapping):              except KeyError:                  pass +    def _scrub_removals(self): +        d = self.data +        self._pending_removals = [k for k in self._pending_removals if k in d] +        self._dirty_len = False +      def __delitem__(self, key): +        self._dirty_len = True          del self.data[ref(key)]      def __getitem__(self, key):          return self.data[ref(key)]      def __len__(self): +        if self._dirty_len and self._pending_removals: +            # self._pending_removals may still contain keys which were +            # explicitly removed, we have to scrub them (see issue #21173). +            self._scrub_removals()          return len(self.data) - len(self._pending_removals)      def __repr__(self): @@ -417,6 +428,7 @@ class WeakKeyDictionary(collections.MutableMapping):          return list(self.data)      def popitem(self): +        self._dirty_len = True          while True:              key, value = self.data.popitem()              o = key() @@ -424,6 +436,7 @@ class WeakKeyDictionary(collections.MutableMapping):                  return o, value      def pop(self, key, *args): +        self._dirty_len = True          return self.data.pop(ref(key), *args)      def setdefault(self, key, default=None): | 
