diff --git a/django/utils/datastructures.py b/django/utils/datastructures.py index dfd3fc69e7..e0835b2cfc 100644 --- a/django/utils/datastructures.py +++ b/django/utils/datastructures.py @@ -62,6 +62,13 @@ class SortedDict(dict): else: self.keyOrder = [key for key, value in data] + def __deepcopy__(self,memo): + from copy import deepcopy + obj = self.__class__() + for k, v in self.items(): + obj[k] = deepcopy(v, memo) + return obj + def __setitem__(self, key, value): dict.__setitem__(self, key, value) if key not in self.keyOrder: @@ -75,6 +82,20 @@ class SortedDict(dict): for k in self.keyOrder: yield k + def pop(self, k, *args): + result = dict.pop(self, k, *args) + try: + self.keyOrder.remove(k) + except ValueError: + # Key wasn't in the dictionary in the first place. No problem. + pass + return result + + def popitem(self): + result = dict.popitem(self) + self.keyOrder.remove(result[0]) + return result + def items(self): return zip(self.keyOrder, self.values()) diff --git a/tests/regressiontests/datastructures/tests.py b/tests/regressiontests/datastructures/tests.py index b58ee79693..d1e21e673c 100644 --- a/tests/regressiontests/datastructures/tests.py +++ b/tests/regressiontests/datastructures/tests.py @@ -54,6 +54,17 @@ True >>> print repr(d) {'one': 'not one', 'two': 'two', 'three': 'three'} +>>> d.pop('one', 'missing') +'not one' +>>> d.pop('one', 'missing') +'missing' + +We don't know which item will be popped in popitem(), so we'll just check that +the number of keys has decreased. +>>> l = len(d) +>>> _ = d.popitem() +>>> l - len(d) +1 Init from sequence of tuples >>> d = SortedDict((