django-rest-framework/rest_framework/utils/encoders.py

71 lines
2.6 KiB
Python
Raw Permalink Normal View History

"""
Helper classes for parsers.
"""
from __future__ import unicode_literals
import datetime
2015-06-25 23:55:51 +03:00
import decimal
import json
import uuid
from django.db.models.query import QuerySet
2015-06-25 23:55:51 +03:00
from django.utils import six, timezone
from django.utils.encoding import force_text
2015-06-25 23:55:51 +03:00
from django.utils.functional import Promise
from rest_framework.compat import coreapi, total_seconds
class JSONEncoder(json.JSONEncoder):
"""
2013-01-15 17:02:53 +04:00
JSONEncoder subclass that knows how to encode date/time/timedelta,
2014-09-12 14:50:20 +04:00
decimal types, generators and other basic python objects.
"""
2014-09-12 14:50:20 +04:00
def default(self, obj):
# For Date Time string spec, see ECMA 262
# http://ecma-international.org/ecma-262/5.1/#sec-15.9.1.15
2014-09-12 14:50:20 +04:00
if isinstance(obj, Promise):
return force_text(obj)
elif isinstance(obj, datetime.datetime):
representation = obj.isoformat()
if representation.endswith('+00:00'):
representation = representation[:-6] + 'Z'
return representation
elif isinstance(obj, datetime.date):
return obj.isoformat()
elif isinstance(obj, datetime.time):
if timezone and timezone.is_aware(obj):
raise ValueError("JSON can't represent timezone-aware times.")
2014-09-12 14:50:20 +04:00
representation = obj.isoformat()
if obj.microsecond:
representation = representation[:12]
return representation
elif isinstance(obj, datetime.timedelta):
return six.text_type(total_seconds(obj))
2014-09-12 14:50:20 +04:00
elif isinstance(obj, decimal.Decimal):
# Serializers will coerce decimals to strings by default.
return float(obj)
elif isinstance(obj, uuid.UUID):
return six.text_type(obj)
2014-09-12 14:50:20 +04:00
elif isinstance(obj, QuerySet):
return tuple(obj)
elif isinstance(obj, six.binary_type):
# Best-effort for binary blobs. See #4187.
return obj.decode('utf-8')
2014-09-12 14:50:20 +04:00
elif hasattr(obj, 'tolist'):
# Numpy arrays and array scalars.
return obj.tolist()
elif hasattr(obj, '__getitem__'):
try:
2014-09-12 14:50:20 +04:00
return dict(obj)
2013-12-09 13:24:10 +04:00
except:
pass
2014-09-12 14:50:20 +04:00
elif hasattr(obj, '__iter__'):
return tuple(item for item in obj)
elif (coreapi is not None) and isinstance(obj, (coreapi.Document, coreapi.Error)):
raise RuntimeError(
'Cannot return a coreapi object from a JSON view. '
'You should be using a schema renderer instead for this view.'
)
2014-09-12 14:50:20 +04:00
return super(JSONEncoder, self).default(obj)