2010-09-28 16:18:39 +08:00
|
|
|
"""
|
|
|
|
Tests for stuff in django.utils.datastructures.
|
|
|
|
"""
|
2021-03-07 04:41:24 +08:00
|
|
|
import collections.abc
|
2011-03-28 10:11:19 +08:00
|
|
|
import copy
|
2020-10-02 23:50:30 +08:00
|
|
|
import pickle
|
2010-09-28 16:18:12 +08:00
|
|
|
|
2014-11-20 01:22:23 +08:00
|
|
|
from django.test import SimpleTestCase
|
2015-01-28 20:35:27 +08:00
|
|
|
from django.utils.datastructures import (
|
2018-05-05 07:37:01 +08:00
|
|
|
CaseInsensitiveMapping,
|
|
|
|
DictWrapper,
|
|
|
|
ImmutableList,
|
|
|
|
MultiValueDict,
|
|
|
|
MultiValueDictKeyError,
|
|
|
|
OrderedSet,
|
2015-01-28 20:35:27 +08:00
|
|
|
)
|
2008-07-13 16:48:18 +08:00
|
|
|
|
2010-09-28 16:18:39 +08:00
|
|
|
|
2014-10-16 18:43:46 +08:00
|
|
|
class OrderedSetTests(SimpleTestCase):
|
2020-02-11 02:41:38 +08:00
|
|
|
def test_init_with_iterable(self):
|
|
|
|
s = OrderedSet([1, 2, 3])
|
|
|
|
self.assertEqual(list(s.dict.keys()), [1, 2, 3])
|
|
|
|
|
|
|
|
def test_remove(self):
|
|
|
|
s = OrderedSet()
|
|
|
|
self.assertEqual(len(s), 0)
|
|
|
|
s.add(1)
|
|
|
|
s.add(2)
|
|
|
|
s.remove(2)
|
|
|
|
self.assertEqual(len(s), 1)
|
|
|
|
self.assertNotIn(2, s)
|
|
|
|
|
|
|
|
def test_discard(self):
|
|
|
|
s = OrderedSet()
|
|
|
|
self.assertEqual(len(s), 0)
|
|
|
|
s.add(1)
|
|
|
|
s.discard(2)
|
|
|
|
self.assertEqual(len(s), 1)
|
|
|
|
|
2021-03-07 04:41:24 +08:00
|
|
|
def test_reversed(self):
|
|
|
|
s = reversed(OrderedSet([1, 2, 3]))
|
|
|
|
self.assertIsInstance(s, collections.abc.Iterator)
|
|
|
|
self.assertEqual(list(s), [3, 2, 1])
|
|
|
|
|
2020-02-11 02:41:38 +08:00
|
|
|
def test_contains(self):
|
|
|
|
s = OrderedSet()
|
|
|
|
self.assertEqual(len(s), 0)
|
|
|
|
s.add(1)
|
|
|
|
self.assertIn(1, s)
|
|
|
|
|
2014-10-16 18:43:46 +08:00
|
|
|
def test_bool(self):
|
|
|
|
# Refs #23664
|
|
|
|
s = OrderedSet()
|
|
|
|
self.assertFalse(s)
|
|
|
|
s.add(1)
|
|
|
|
self.assertTrue(s)
|
|
|
|
|
2015-07-10 02:00:36 +08:00
|
|
|
def test_len(self):
|
|
|
|
s = OrderedSet()
|
|
|
|
self.assertEqual(len(s), 0)
|
|
|
|
s.add(1)
|
|
|
|
s.add(2)
|
|
|
|
s.add(2)
|
|
|
|
self.assertEqual(len(s), 2)
|
|
|
|
|
2021-03-10 16:29:05 +08:00
|
|
|
def test_repr(self):
|
|
|
|
self.assertEqual(repr(OrderedSet()), "OrderedSet()")
|
|
|
|
self.assertEqual(repr(OrderedSet([2, 3, 2, 1])), "OrderedSet([2, 3, 1])")
|
|
|
|
|
2014-10-16 18:43:46 +08:00
|
|
|
|
2011-08-13 08:42:08 +08:00
|
|
|
class MultiValueDictTests(SimpleTestCase):
|
2020-10-02 23:50:30 +08:00
|
|
|
def test_repr(self):
|
|
|
|
d = MultiValueDict({"key": "value"})
|
|
|
|
self.assertEqual(repr(d), "<MultiValueDict: {'key': 'value'}>")
|
|
|
|
|
2010-09-28 16:18:39 +08:00
|
|
|
def test_multivaluedict(self):
|
2020-10-02 23:50:30 +08:00
|
|
|
d = MultiValueDict(
|
|
|
|
{"name": ["Adrian", "Simon"], "position": ["Developer"], "empty": []}
|
|
|
|
)
|
2011-03-03 23:04:39 +08:00
|
|
|
self.assertEqual(d["name"], "Simon")
|
|
|
|
self.assertEqual(d.get("name"), "Simon")
|
|
|
|
self.assertEqual(d.getlist("name"), ["Adrian", "Simon"])
|
2013-12-13 04:23:24 +08:00
|
|
|
self.assertEqual(
|
2020-10-02 23:50:30 +08:00
|
|
|
list(d.items()),
|
|
|
|
[("name", "Simon"), ("position", "Developer"), ("empty", [])],
|
2013-12-13 04:23:24 +08:00
|
|
|
)
|
|
|
|
self.assertEqual(
|
2020-10-02 23:50:30 +08:00
|
|
|
list(d.lists()),
|
|
|
|
[("name", ["Adrian", "Simon"]), ("position", ["Developer"]), ("empty", [])],
|
2013-12-13 04:23:24 +08:00
|
|
|
)
|
2020-02-05 04:58:07 +08:00
|
|
|
with self.assertRaisesMessage(MultiValueDictKeyError, "'lastname'"):
|
2016-01-18 16:45:45 +08:00
|
|
|
d.__getitem__("lastname")
|
2020-10-02 23:50:30 +08:00
|
|
|
self.assertIsNone(d.get("empty"))
|
|
|
|
self.assertEqual(d.get("empty", "nonexistent"), "nonexistent")
|
2016-06-17 02:19:18 +08:00
|
|
|
self.assertIsNone(d.get("lastname"))
|
2011-03-03 23:04:39 +08:00
|
|
|
self.assertEqual(d.get("lastname", "nonexistent"), "nonexistent")
|
|
|
|
self.assertEqual(d.getlist("lastname"), [])
|
2019-01-03 07:18:19 +08:00
|
|
|
self.assertEqual(
|
|
|
|
d.getlist("doesnotexist", ["Adrian", "Simon"]), ["Adrian", "Simon"]
|
|
|
|
)
|
2010-09-28 16:18:39 +08:00
|
|
|
d.setlist("lastname", ["Holovaty", "Willison"])
|
2011-03-03 23:04:39 +08:00
|
|
|
self.assertEqual(d.getlist("lastname"), ["Holovaty", "Willison"])
|
2020-10-02 23:50:30 +08:00
|
|
|
self.assertEqual(list(d.values()), ["Simon", "Developer", [], "Willison"])
|
2010-09-28 16:18:39 +08:00
|
|
|
|
2023-02-14 08:24:37 +08:00
|
|
|
d.setlistdefault("lastname", ["Doe"])
|
|
|
|
self.assertEqual(d.getlist("lastname"), ["Holovaty", "Willison"])
|
|
|
|
d.setlistdefault("newkey", ["Doe"])
|
|
|
|
self.assertEqual(d.getlist("newkey"), ["Doe"])
|
|
|
|
|
2012-02-10 02:41:20 +08:00
|
|
|
def test_appendlist(self):
|
|
|
|
d = MultiValueDict()
|
|
|
|
d.appendlist("name", "Adrian")
|
|
|
|
d.appendlist("name", "Simon")
|
|
|
|
self.assertEqual(d.getlist("name"), ["Adrian", "Simon"])
|
|
|
|
|
2010-10-28 04:39:20 +08:00
|
|
|
def test_copy(self):
|
2011-03-28 22:20:15 +08:00
|
|
|
for copy_func in [copy.copy, lambda d: d.copy()]:
|
2020-10-02 23:50:30 +08:00
|
|
|
with self.subTest(copy_func):
|
|
|
|
d1 = MultiValueDict({"developers": ["Carl", "Fred"]})
|
|
|
|
self.assertEqual(d1["developers"], "Fred")
|
|
|
|
d2 = copy_func(d1)
|
|
|
|
d2.update({"developers": "Groucho"})
|
|
|
|
self.assertEqual(d2["developers"], "Groucho")
|
|
|
|
self.assertEqual(d1["developers"], "Fred")
|
|
|
|
|
|
|
|
d1 = MultiValueDict({"key": [[]]})
|
|
|
|
self.assertEqual(d1["key"], [])
|
|
|
|
d2 = copy_func(d1)
|
|
|
|
d2["key"].append("Penguin")
|
|
|
|
self.assertEqual(d1["key"], ["Penguin"])
|
|
|
|
self.assertEqual(d2["key"], ["Penguin"])
|
|
|
|
|
|
|
|
def test_deepcopy(self):
|
|
|
|
d1 = MultiValueDict({"a": [[123]]})
|
|
|
|
d2 = copy.copy(d1)
|
|
|
|
d3 = copy.deepcopy(d1)
|
|
|
|
self.assertIs(d1["a"], d2["a"])
|
|
|
|
self.assertIsNot(d1["a"], d3["a"])
|
|
|
|
|
|
|
|
def test_pickle(self):
|
|
|
|
x = MultiValueDict({"a": ["1", "2"], "b": ["3"]})
|
|
|
|
self.assertEqual(x, pickle.loads(pickle.dumps(x)))
|
2010-10-28 04:39:20 +08:00
|
|
|
|
2011-06-17 00:34:55 +08:00
|
|
|
def test_dict_translation(self):
|
|
|
|
mvd = MultiValueDict(
|
|
|
|
{
|
|
|
|
"devs": ["Bob", "Joe"],
|
|
|
|
"pm": ["Rory"],
|
|
|
|
}
|
|
|
|
)
|
|
|
|
d = mvd.dict()
|
2020-10-02 23:50:30 +08:00
|
|
|
self.assertEqual(list(d), list(mvd))
|
2017-05-28 07:08:46 +08:00
|
|
|
for key in mvd:
|
2011-06-17 00:34:55 +08:00
|
|
|
self.assertEqual(d[key], mvd[key])
|
|
|
|
|
|
|
|
self.assertEqual({}, MultiValueDict().dict())
|
|
|
|
|
2016-09-10 17:25:42 +08:00
|
|
|
def test_getlist_doesnt_mutate(self):
|
|
|
|
x = MultiValueDict({"a": ["1", "2"], "b": ["3"]})
|
|
|
|
values = x.getlist("a")
|
|
|
|
values += x.getlist("b")
|
|
|
|
self.assertEqual(x.getlist("a"), ["1", "2"])
|
|
|
|
|
|
|
|
def test_internal_getlist_does_mutate(self):
|
|
|
|
x = MultiValueDict({"a": ["1", "2"], "b": ["3"]})
|
|
|
|
values = x._getlist("a")
|
|
|
|
values += x._getlist("b")
|
|
|
|
self.assertEqual(x._getlist("a"), ["1", "2", "3"])
|
|
|
|
|
|
|
|
def test_getlist_default(self):
|
|
|
|
x = MultiValueDict({"a": [1]})
|
|
|
|
MISSING = object()
|
|
|
|
values = x.getlist("b", default=MISSING)
|
|
|
|
self.assertIs(values, MISSING)
|
|
|
|
|
2016-12-10 04:31:52 +08:00
|
|
|
def test_getlist_none_empty_values(self):
|
|
|
|
x = MultiValueDict({"a": None, "b": []})
|
|
|
|
self.assertIsNone(x.getlist("a"))
|
|
|
|
self.assertEqual(x.getlist("b"), [])
|
|
|
|
|
2020-10-02 23:50:30 +08:00
|
|
|
def test_setitem(self):
|
|
|
|
x = MultiValueDict({"a": [1, 2]})
|
|
|
|
x["a"] = 3
|
|
|
|
self.assertEqual(list(x.lists()), [("a", [3])])
|
|
|
|
|
|
|
|
def test_setdefault(self):
|
|
|
|
x = MultiValueDict({"a": [1, 2]})
|
|
|
|
a = x.setdefault("a", 3)
|
|
|
|
b = x.setdefault("b", 3)
|
|
|
|
self.assertEqual(a, 2)
|
|
|
|
self.assertEqual(b, 3)
|
|
|
|
self.assertEqual(list(x.lists()), [("a", [1, 2]), ("b", [3])])
|
|
|
|
|
|
|
|
def test_update_too_many_args(self):
|
|
|
|
x = MultiValueDict({"a": []})
|
|
|
|
msg = "update expected at most 1 argument, got 2"
|
|
|
|
with self.assertRaisesMessage(TypeError, msg):
|
|
|
|
x.update(1, 2)
|
|
|
|
|
|
|
|
def test_update_no_args(self):
|
|
|
|
x = MultiValueDict({"a": []})
|
|
|
|
x.update()
|
|
|
|
self.assertEqual(list(x.lists()), [("a", [])])
|
|
|
|
|
|
|
|
def test_update_dict_arg(self):
|
|
|
|
x = MultiValueDict({"a": [1], "b": [2], "c": [3]})
|
|
|
|
x.update({"a": 4, "b": 5})
|
|
|
|
self.assertEqual(list(x.lists()), [("a", [1, 4]), ("b", [2, 5]), ("c", [3])])
|
|
|
|
|
|
|
|
def test_update_multivaluedict_arg(self):
|
|
|
|
x = MultiValueDict({"a": [1], "b": [2], "c": [3]})
|
|
|
|
x.update(MultiValueDict({"a": [4], "b": [5]}))
|
|
|
|
self.assertEqual(list(x.lists()), [("a", [1, 4]), ("b", [2, 5]), ("c", [3])])
|
|
|
|
|
|
|
|
def test_update_kwargs(self):
|
|
|
|
x = MultiValueDict({"a": [1], "b": [2], "c": [3]})
|
|
|
|
x.update(a=4, b=5)
|
|
|
|
self.assertEqual(list(x.lists()), [("a", [1, 4]), ("b", [2, 5]), ("c", [3])])
|
|
|
|
|
2020-10-05 22:57:47 +08:00
|
|
|
def test_update_with_empty_iterable(self):
|
|
|
|
for value in ["", b"", (), [], set(), {}]:
|
|
|
|
d = MultiValueDict()
|
|
|
|
d.update(value)
|
|
|
|
self.assertEqual(d, MultiValueDict())
|
|
|
|
|
|
|
|
def test_update_with_iterable_of_pairs(self):
|
|
|
|
for value in [(("a", 1),), [("a", 1)], {("a", 1)}]:
|
|
|
|
d = MultiValueDict()
|
|
|
|
d.update(value)
|
|
|
|
self.assertEqual(d, MultiValueDict({"a": [1]}))
|
|
|
|
|
|
|
|
def test_update_raises_correct_exceptions(self):
|
|
|
|
# MultiValueDict.update() raises equivalent exceptions to
|
|
|
|
# dict.update().
|
|
|
|
# Non-iterable values raise TypeError.
|
|
|
|
for value in [None, True, False, 123, 123.45]:
|
|
|
|
with self.subTest(value), self.assertRaises(TypeError):
|
|
|
|
MultiValueDict().update(value)
|
|
|
|
# Iterables of objects that cannot be unpacked raise TypeError.
|
|
|
|
for value in [b"123", b"abc", (1, 2, 3), [1, 2, 3], {1, 2, 3}]:
|
|
|
|
with self.subTest(value), self.assertRaises(TypeError):
|
|
|
|
MultiValueDict().update(value)
|
|
|
|
# Iterables of unpackable objects with incorrect number of items raise
|
|
|
|
# ValueError.
|
|
|
|
for value in ["123", "abc", ("a", "b", "c"), ["a", "b", "c"], {"a", "b", "c"}]:
|
|
|
|
with self.subTest(value), self.assertRaises(ValueError):
|
|
|
|
MultiValueDict().update(value)
|
|
|
|
|
2010-09-28 16:18:39 +08:00
|
|
|
|
2011-08-13 08:42:08 +08:00
|
|
|
class ImmutableListTests(SimpleTestCase):
|
2010-09-28 16:18:39 +08:00
|
|
|
def test_sort(self):
|
|
|
|
d = ImmutableList(range(10))
|
|
|
|
|
|
|
|
# AttributeError: ImmutableList object is immutable.
|
2016-01-04 16:50:08 +08:00
|
|
|
with self.assertRaisesMessage(
|
|
|
|
AttributeError, "ImmutableList object is immutable."
|
|
|
|
):
|
|
|
|
d.sort()
|
2010-09-28 16:18:39 +08:00
|
|
|
|
2011-03-03 23:04:39 +08:00
|
|
|
self.assertEqual(repr(d), "(0, 1, 2, 3, 4, 5, 6, 7, 8, 9)")
|
2010-09-28 16:18:39 +08:00
|
|
|
|
|
|
|
def test_custom_warning(self):
|
|
|
|
d = ImmutableList(range(10), warning="Object is immutable!")
|
|
|
|
|
2011-03-03 23:04:39 +08:00
|
|
|
self.assertEqual(d[1], 1)
|
2010-09-28 16:18:39 +08:00
|
|
|
|
|
|
|
# AttributeError: Object is immutable!
|
2016-01-04 16:50:08 +08:00
|
|
|
with self.assertRaisesMessage(AttributeError, "Object is immutable!"):
|
|
|
|
d.__setitem__(1, "test")
|
2010-09-28 16:18:39 +08:00
|
|
|
|
|
|
|
|
2011-08-13 08:42:08 +08:00
|
|
|
class DictWrapperTests(SimpleTestCase):
|
2010-09-28 16:18:39 +08:00
|
|
|
def test_dictwrapper(self):
|
2016-01-24 00:47:07 +08:00
|
|
|
def f(x):
|
|
|
|
return "*%s" % x
|
2022-02-04 03:24:19 +08:00
|
|
|
|
2010-09-28 16:18:39 +08:00
|
|
|
d = DictWrapper({"a": "a"}, f, "xx_")
|
2013-12-13 04:23:24 +08:00
|
|
|
self.assertEqual(
|
|
|
|
"Normal: %(a)s. Modified: %(xx_a)s" % d, "Normal: a. Modified: *a"
|
|
|
|
)
|
2018-05-05 07:37:01 +08:00
|
|
|
|
|
|
|
|
|
|
|
class CaseInsensitiveMappingTests(SimpleTestCase):
|
|
|
|
def setUp(self):
|
|
|
|
self.dict1 = CaseInsensitiveMapping(
|
|
|
|
{
|
|
|
|
"Accept": "application/json",
|
|
|
|
"content-type": "text/html",
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
def test_create_with_invalid_values(self):
|
|
|
|
msg = "dictionary update sequence element #1 has length 4; 2 is required"
|
|
|
|
with self.assertRaisesMessage(ValueError, msg):
|
|
|
|
CaseInsensitiveMapping([("Key1", "Val1"), "Key2"])
|
|
|
|
|
|
|
|
def test_create_with_invalid_key(self):
|
|
|
|
msg = "Element key 1 invalid, only strings are allowed"
|
|
|
|
with self.assertRaisesMessage(ValueError, msg):
|
|
|
|
CaseInsensitiveMapping([(1, "2")])
|
|
|
|
|
|
|
|
def test_list(self):
|
2019-02-09 18:07:04 +08:00
|
|
|
self.assertEqual(list(self.dict1), ["Accept", "content-type"])
|
2018-05-05 07:37:01 +08:00
|
|
|
|
|
|
|
def test_dict(self):
|
|
|
|
self.assertEqual(
|
|
|
|
dict(self.dict1),
|
|
|
|
{"Accept": "application/json", "content-type": "text/html"},
|
|
|
|
)
|
|
|
|
|
|
|
|
def test_repr(self):
|
|
|
|
dict1 = CaseInsensitiveMapping({"Accept": "application/json"})
|
|
|
|
dict2 = CaseInsensitiveMapping({"content-type": "text/html"})
|
|
|
|
self.assertEqual(repr(dict1), repr({"Accept": "application/json"}))
|
|
|
|
self.assertEqual(repr(dict2), repr({"content-type": "text/html"}))
|
|
|
|
|
|
|
|
def test_str(self):
|
|
|
|
dict1 = CaseInsensitiveMapping({"Accept": "application/json"})
|
|
|
|
dict2 = CaseInsensitiveMapping({"content-type": "text/html"})
|
|
|
|
self.assertEqual(str(dict1), str({"Accept": "application/json"}))
|
|
|
|
self.assertEqual(str(dict2), str({"content-type": "text/html"}))
|
|
|
|
|
|
|
|
def test_equal(self):
|
|
|
|
self.assertEqual(
|
|
|
|
self.dict1, {"Accept": "application/json", "content-type": "text/html"}
|
|
|
|
)
|
|
|
|
self.assertNotEqual(
|
|
|
|
self.dict1, {"accept": "application/jso", "Content-Type": "text/html"}
|
2022-02-04 03:24:19 +08:00
|
|
|
)
|
2018-05-05 07:37:01 +08:00
|
|
|
self.assertNotEqual(self.dict1, "string")
|
|
|
|
|
|
|
|
def test_items(self):
|
|
|
|
other = {"Accept": "application/json", "content-type": "text/html"}
|
|
|
|
self.assertEqual(sorted(self.dict1.items()), sorted(other.items()))
|
|
|
|
|
|
|
|
def test_copy(self):
|
|
|
|
copy = self.dict1.copy()
|
|
|
|
self.assertIs(copy, self.dict1)
|
|
|
|
self.assertEqual(copy, self.dict1)
|
|
|
|
|
|
|
|
def test_getitem(self):
|
|
|
|
self.assertEqual(self.dict1["Accept"], "application/json")
|
|
|
|
self.assertEqual(self.dict1["accept"], "application/json")
|
|
|
|
self.assertEqual(self.dict1["aCCept"], "application/json")
|
|
|
|
self.assertEqual(self.dict1["content-type"], "text/html")
|
|
|
|
self.assertEqual(self.dict1["Content-Type"], "text/html")
|
|
|
|
self.assertEqual(self.dict1["Content-type"], "text/html")
|
|
|
|
|
|
|
|
def test_in(self):
|
|
|
|
self.assertIn("Accept", self.dict1)
|
|
|
|
self.assertIn("accept", self.dict1)
|
|
|
|
self.assertIn("aCCept", self.dict1)
|
|
|
|
self.assertIn("content-type", self.dict1)
|
|
|
|
self.assertIn("Content-Type", self.dict1)
|
|
|
|
|
|
|
|
def test_del(self):
|
|
|
|
self.assertIn("Accept", self.dict1)
|
|
|
|
msg = "'CaseInsensitiveMapping' object does not support item deletion"
|
|
|
|
with self.assertRaisesMessage(TypeError, msg):
|
|
|
|
del self.dict1["Accept"]
|
|
|
|
self.assertIn("Accept", self.dict1)
|
|
|
|
|
|
|
|
def test_set(self):
|
|
|
|
self.assertEqual(len(self.dict1), 2)
|
|
|
|
msg = "'CaseInsensitiveMapping' object does not support item assignment"
|
|
|
|
with self.assertRaisesMessage(TypeError, msg):
|
|
|
|
self.dict1["New Key"] = 1
|
|
|
|
self.assertEqual(len(self.dict1), 2)
|