mirror of
				https://github.com/encode/django-rest-framework.git
				synced 2025-10-25 13:11:26 +03:00 
			
		
		
		
	* fix OpenAPIRenderer for timedelta * added test for rendering openapi with timedelta * fix OpenAPIRenderer for timedelta * added test for rendering openapi with timedelta * Removed usage of field.choices that triggered full table load (#8950) Removed the `{{ field.choices|yesno:",disabled" }}` block because this triggers the loading of full database table worth of objects just to determine whether the multi-select widget should be set as disabled or not. Since this "disabled" marking feature is not present in the normal select field, then I propose to remove it also from the multi-select. * Added Deprecation Warnings for CoreAPI (#7519) * Added Deprecation Warnings for CoreAPI * Bumped removal to DRF315 * Update rest_framework/__init__.py * Update rest_framework/filters.py * Update rest_framework/filters.py * Update tests/schemas/test_coreapi.py * Update rest_framework/filters.py * Update rest_framework/filters.py * Update tests/schemas/test_coreapi.py * Update tests/schemas/test_coreapi.py * Update setup.cfg * Update rest_framework/pagination.py --------- Co-authored-by: Asif Saif Uddin <auvipy@gmail.com> * Update copy right timeline * Fix NamespaceVersioning ignoring DEFAULT_VERSION on non-None namespaces (#7278) * Fix the case where if the namespace is not None and there's no match, NamespaceVersioning always raises NotFound even if DEFAULT_VERSION is set or None is in ALLOWED_VERSIONS * Add test cases * fix OpenAPIRenderer for timedelta * added test for rendering openapi with timedelta * added testcase for rendering yaml with minvalidator for duration field (timedelta) --------- Co-authored-by: Rizwan Shaikh <rshaikh@ces-ltd.com> Co-authored-by: Lenno Nagel <lenno@namespace.ee> Co-authored-by: David Smith <39445562+smithdc1@users.noreply.github.com> Co-authored-by: Asif Saif Uddin <auvipy@gmail.com> Co-authored-by: Konstantin Kuchkov <konstantin.kuchkov@gmail.com>
		
			
				
	
	
		
			79 lines
		
	
	
		
			2.8 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			79 lines
		
	
	
		
			2.8 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| """
 | |
| Helper classes for parsers.
 | |
| """
 | |
| 
 | |
| import contextlib
 | |
| import datetime
 | |
| import decimal
 | |
| import json  # noqa
 | |
| import uuid
 | |
| 
 | |
| from django.db.models.query import QuerySet
 | |
| from django.utils import timezone
 | |
| from django.utils.encoding import force_str
 | |
| from django.utils.functional import Promise
 | |
| 
 | |
| from rest_framework.compat import coreapi
 | |
| 
 | |
| 
 | |
| class JSONEncoder(json.JSONEncoder):
 | |
|     """
 | |
|     JSONEncoder subclass that knows how to encode date/time/timedelta,
 | |
|     decimal types, generators and other basic python objects.
 | |
|     """
 | |
|     def default(self, obj):
 | |
|         # For Date Time string spec, see ECMA 262
 | |
|         # https://ecma-international.org/ecma-262/5.1/#sec-15.9.1.15
 | |
|         if isinstance(obj, Promise):
 | |
|             return force_str(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.")
 | |
|             representation = obj.isoformat()
 | |
|             return representation
 | |
|         elif isinstance(obj, datetime.timedelta):
 | |
|             return str(obj.total_seconds())
 | |
|         elif isinstance(obj, decimal.Decimal):
 | |
|             # Serializers will coerce decimals to strings by default.
 | |
|             return float(obj)
 | |
|         elif isinstance(obj, uuid.UUID):
 | |
|             return str(obj)
 | |
|         elif isinstance(obj, QuerySet):
 | |
|             return tuple(obj)
 | |
|         elif isinstance(obj, bytes):
 | |
|             # Best-effort for binary blobs. See #4187.
 | |
|             return obj.decode()
 | |
|         elif hasattr(obj, 'tolist'):
 | |
|             # Numpy arrays and array scalars.
 | |
|             return obj.tolist()
 | |
|         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.'
 | |
|             )
 | |
|         elif hasattr(obj, '__getitem__'):
 | |
|             cls = (list if isinstance(obj, (list, tuple)) else dict)
 | |
|             with contextlib.suppress(Exception):
 | |
|                 return cls(obj)
 | |
|         elif hasattr(obj, '__iter__'):
 | |
|             return tuple(item for item in obj)
 | |
|         return super().default(obj)
 | |
| 
 | |
| 
 | |
| class CustomScalar:
 | |
|     """
 | |
|     CustomScalar that knows how to encode timedelta that renderer
 | |
|     can understand.
 | |
|     """
 | |
|     @classmethod
 | |
|     def represent_timedelta(cls, dumper, data):
 | |
|         value = str(data.total_seconds())
 | |
|         return dumper.represent_scalar('tag:yaml.org,2002:str', value)
 |