2018-10-16 23:02:36 +08:00
|
|
|
from django.utils.itercompat import is_iterable
|
|
|
|
|
|
|
|
|
2018-10-16 23:01:31 +08:00
|
|
|
def make_hashable(value):
|
2020-10-05 21:13:14 +08:00
|
|
|
"""
|
|
|
|
Attempt to make value hashable or raise a TypeError if it fails.
|
|
|
|
|
|
|
|
The returned value should generate the same hash for equal values.
|
|
|
|
"""
|
2018-10-16 23:01:31 +08:00
|
|
|
if isinstance(value, dict):
|
|
|
|
return tuple([
|
|
|
|
(key, make_hashable(nested_value))
|
2020-10-05 21:13:14 +08:00
|
|
|
for key, nested_value in sorted(value.items())
|
2018-10-16 23:01:31 +08:00
|
|
|
])
|
2018-10-16 23:02:36 +08:00
|
|
|
# Try hash to avoid converting a hashable iterable (e.g. string, frozenset)
|
|
|
|
# to a tuple.
|
|
|
|
try:
|
|
|
|
hash(value)
|
|
|
|
except TypeError:
|
|
|
|
if is_iterable(value):
|
|
|
|
return tuple(map(make_hashable, value))
|
|
|
|
# Non-hashable, non-iterable.
|
|
|
|
raise
|
2018-10-16 23:01:31 +08:00
|
|
|
return value
|