mirror of
				https://github.com/django/django.git
				synced 2025-10-25 22:56:12 +00:00 
			
		
		
		
	Fixed #29522 -- Refactored the Deserializer functions to classes.
Co-authored-by: Emad Mokhtar <emad.mokhtar@veneficus.nl>
This commit is contained in:
		
							
								
								
									
										2
									
								
								AUTHORS
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								AUTHORS
									
									
									
									
									
								
							| @@ -68,7 +68,7 @@ answer newbie questions, and generally made Django that much better: | |||||||
|     Aljaž Košir <aljazkosir5@gmail.com> |     Aljaž Košir <aljazkosir5@gmail.com> | ||||||
|     Aljosa Mohorovic <aljosa.mohorovic@gmail.com> |     Aljosa Mohorovic <aljosa.mohorovic@gmail.com> | ||||||
|     Alokik Vijay <alokik.roe@gmail.com> |     Alokik Vijay <alokik.roe@gmail.com> | ||||||
|     Amir Karimi <amk9978@gmail.com> |     Amir Karimi <https://github.com/amk9978> | ||||||
|     Amit Chakradeo <https://amit.chakradeo.net/> |     Amit Chakradeo <https://amit.chakradeo.net/> | ||||||
|     Amit Ramon <amit.ramon@gmail.com> |     Amit Ramon <amit.ramon@gmail.com> | ||||||
|     Amit Upadhyay <http://www.amitu.com/blog/> |     Amit Upadhyay <http://www.amitu.com/blog/> | ||||||
|   | |||||||
| @@ -59,19 +59,27 @@ class Serializer(PythonSerializer): | |||||||
|         return super(PythonSerializer, self).getvalue() |         return super(PythonSerializer, self).getvalue() | ||||||
|  |  | ||||||
|  |  | ||||||
| def Deserializer(stream_or_string, **options): | class Deserializer(PythonDeserializer): | ||||||
|     """Deserialize a stream or string of JSON data.""" |     """Deserialize a stream or string of JSON data.""" | ||||||
|     if not isinstance(stream_or_string, (bytes, str)): |  | ||||||
|         stream_or_string = stream_or_string.read() |     def __init__(self, stream_or_string, **options): | ||||||
|     if isinstance(stream_or_string, bytes): |         if not isinstance(stream_or_string, (bytes, str)): | ||||||
|         stream_or_string = stream_or_string.decode() |             stream_or_string = stream_or_string.read() | ||||||
|     try: |         if isinstance(stream_or_string, bytes): | ||||||
|         objects = json.loads(stream_or_string) |             stream_or_string = stream_or_string.decode() | ||||||
|         yield from PythonDeserializer(objects, **options) |         try: | ||||||
|     except (GeneratorExit, DeserializationError): |             objects = json.loads(stream_or_string) | ||||||
|         raise |         except Exception as exc: | ||||||
|     except Exception as exc: |             raise DeserializationError() from exc | ||||||
|         raise DeserializationError() from exc |         super().__init__(objects, **options) | ||||||
|  |  | ||||||
|  |     def _handle_object(self, obj): | ||||||
|  |         try: | ||||||
|  |             yield from super()._handle_object(obj) | ||||||
|  |         except (GeneratorExit, DeserializationError): | ||||||
|  |             raise | ||||||
|  |         except Exception as exc: | ||||||
|  |             raise DeserializationError(f"Error deserializing object: {exc}") from exc | ||||||
|  |  | ||||||
|  |  | ||||||
| class DjangoJSONEncoder(json.JSONEncoder): | class DjangoJSONEncoder(json.JSONEncoder): | ||||||
|   | |||||||
| @@ -39,19 +39,30 @@ class Serializer(PythonSerializer): | |||||||
|         return super(PythonSerializer, self).getvalue() |         return super(PythonSerializer, self).getvalue() | ||||||
|  |  | ||||||
|  |  | ||||||
| def Deserializer(stream_or_string, **options): | class Deserializer(PythonDeserializer): | ||||||
|     """Deserialize a stream or string of JSON data.""" |     """Deserialize a stream or string of JSON data.""" | ||||||
|     if isinstance(stream_or_string, bytes): |  | ||||||
|         stream_or_string = stream_or_string.decode() |  | ||||||
|     if isinstance(stream_or_string, (bytes, str)): |  | ||||||
|         stream_or_string = stream_or_string.split("\n") |  | ||||||
|  |  | ||||||
|     for line in stream_or_string: |     def __init__(self, stream_or_string, **options): | ||||||
|         if not line.strip(): |         if isinstance(stream_or_string, bytes): | ||||||
|             continue |             stream_or_string = stream_or_string.decode() | ||||||
|  |         if isinstance(stream_or_string, str): | ||||||
|  |             stream_or_string = stream_or_string.splitlines() | ||||||
|  |         super().__init__(Deserializer._get_lines(stream_or_string), **options) | ||||||
|  |  | ||||||
|  |     def _handle_object(self, obj): | ||||||
|         try: |         try: | ||||||
|             yield from PythonDeserializer([json.loads(line)], **options) |             yield from super()._handle_object(obj) | ||||||
|         except (GeneratorExit, DeserializationError): |         except (GeneratorExit, DeserializationError): | ||||||
|             raise |             raise | ||||||
|         except Exception as exc: |         except Exception as exc: | ||||||
|             raise DeserializationError() from exc |             raise DeserializationError(f"Error deserializing object: {exc}") from exc | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def _get_lines(stream): | ||||||
|  |         for line in stream: | ||||||
|  |             if not line.strip(): | ||||||
|  |                 continue | ||||||
|  |             try: | ||||||
|  |                 yield json.loads(line) | ||||||
|  |             except Exception as exc: | ||||||
|  |                 raise DeserializationError() from exc | ||||||
|   | |||||||
| @@ -96,45 +96,60 @@ class Serializer(base.Serializer): | |||||||
|         return self.objects |         return self.objects | ||||||
|  |  | ||||||
|  |  | ||||||
| def Deserializer( | class Deserializer(base.Deserializer): | ||||||
|     object_list, *, using=DEFAULT_DB_ALIAS, ignorenonexistent=False, **options |  | ||||||
| ): |  | ||||||
|     """ |     """ | ||||||
|     Deserialize simple Python objects back into Django ORM instances. |     Deserialize simple Python objects back into Django ORM instances. | ||||||
|  |  | ||||||
|     It's expected that you pass the Python objects themselves (instead of a |     It's expected that you pass the Python objects themselves (instead of a | ||||||
|     stream or a string) to the constructor |     stream or a string) to the constructor | ||||||
|     """ |     """ | ||||||
|     handle_forward_references = options.pop("handle_forward_references", False) |  | ||||||
|     field_names_cache = {}  # Model: <list of field_names> |  | ||||||
|  |  | ||||||
|     for d in object_list: |     def __init__( | ||||||
|         # Look up the model and starting build a dict of data for it. |         self, object_list, *, using=DEFAULT_DB_ALIAS, ignorenonexistent=False, **options | ||||||
|         try: |     ): | ||||||
|             Model = _get_model(d["model"]) |         super().__init__(object_list, **options) | ||||||
|         except base.DeserializationError: |         self.handle_forward_references = options.pop("handle_forward_references", False) | ||||||
|             if ignorenonexistent: |         self.using = using | ||||||
|                 continue |         self.ignorenonexistent = ignorenonexistent | ||||||
|             else: |         self.field_names_cache = {}  # Model: <list of field_names> | ||||||
|                 raise |         self._iterator = None | ||||||
|  |  | ||||||
|  |     def __iter__(self): | ||||||
|  |         for obj in self.stream: | ||||||
|  |             yield from self._handle_object(obj) | ||||||
|  |  | ||||||
|  |     def __next__(self): | ||||||
|  |         if self._iterator is None: | ||||||
|  |             self._iterator = iter(self) | ||||||
|  |         return next(self._iterator) | ||||||
|  |  | ||||||
|  |     def _handle_object(self, obj): | ||||||
|         data = {} |         data = {} | ||||||
|         if "pk" in d: |  | ||||||
|             try: |  | ||||||
|                 data[Model._meta.pk.attname] = Model._meta.pk.to_python(d.get("pk")) |  | ||||||
|             except Exception as e: |  | ||||||
|                 raise base.DeserializationError.WithData( |  | ||||||
|                     e, d["model"], d.get("pk"), None |  | ||||||
|                 ) |  | ||||||
|         m2m_data = {} |         m2m_data = {} | ||||||
|         deferred_fields = {} |         deferred_fields = {} | ||||||
|  |  | ||||||
|         if Model not in field_names_cache: |         # Look up the model and starting build a dict of data for it. | ||||||
|             field_names_cache[Model] = {f.name for f in Model._meta.get_fields()} |         try: | ||||||
|         field_names = field_names_cache[Model] |             Model = self._get_model_from_node(obj["model"]) | ||||||
|  |         except base.DeserializationError: | ||||||
|  |             if self.ignorenonexistent: | ||||||
|  |                 return | ||||||
|  |             raise | ||||||
|  |         if "pk" in obj: | ||||||
|  |             try: | ||||||
|  |                 data[Model._meta.pk.attname] = Model._meta.pk.to_python(obj.get("pk")) | ||||||
|  |             except Exception as e: | ||||||
|  |                 raise base.DeserializationError.WithData( | ||||||
|  |                     e, obj["model"], obj.get("pk"), None | ||||||
|  |                 ) | ||||||
|  |  | ||||||
|  |         if Model not in self.field_names_cache: | ||||||
|  |             self.field_names_cache[Model] = {f.name for f in Model._meta.get_fields()} | ||||||
|  |         field_names = self.field_names_cache[Model] | ||||||
|  |  | ||||||
|         # Handle each field |         # Handle each field | ||||||
|         for field_name, field_value in d["fields"].items(): |         for field_name, field_value in obj["fields"].items(): | ||||||
|             if ignorenonexistent and field_name not in field_names: |             if self.ignorenonexistent and field_name not in field_names: | ||||||
|                 # skip fields no longer on model |                 # skip fields no longer on model | ||||||
|                 continue |                 continue | ||||||
|  |  | ||||||
| @@ -145,51 +160,59 @@ def Deserializer( | |||||||
|                 field.remote_field, models.ManyToManyRel |                 field.remote_field, models.ManyToManyRel | ||||||
|             ): |             ): | ||||||
|                 try: |                 try: | ||||||
|                     values = base.deserialize_m2m_values( |                     values = self._handle_m2m_field_node(field, field_value) | ||||||
|                         field, field_value, using, handle_forward_references |                     if values == base.DEFER_FIELD: | ||||||
|                     ) |                         deferred_fields[field] = field_value | ||||||
|  |                     else: | ||||||
|  |                         m2m_data[field.name] = values | ||||||
|                 except base.M2MDeserializationError as e: |                 except base.M2MDeserializationError as e: | ||||||
|                     raise base.DeserializationError.WithData( |                     raise base.DeserializationError.WithData( | ||||||
|                         e.original_exc, d["model"], d.get("pk"), e.pk |                         e.original_exc, obj["model"], obj.get("pk"), e.pk | ||||||
|                     ) |                     ) | ||||||
|                 if values == base.DEFER_FIELD: |  | ||||||
|                     deferred_fields[field] = field_value |  | ||||||
|                 else: |  | ||||||
|                     m2m_data[field.name] = values |  | ||||||
|             # Handle FK fields |             # Handle FK fields | ||||||
|             elif field.remote_field and isinstance( |             elif field.remote_field and isinstance( | ||||||
|                 field.remote_field, models.ManyToOneRel |                 field.remote_field, models.ManyToOneRel | ||||||
|             ): |             ): | ||||||
|                 try: |                 try: | ||||||
|                     value = base.deserialize_fk_value( |                     value = self._handle_fk_field_node(field, field_value) | ||||||
|                         field, field_value, using, handle_forward_references |                     if value == base.DEFER_FIELD: | ||||||
|                     ) |                         deferred_fields[field] = field_value | ||||||
|  |                     else: | ||||||
|  |                         data[field.attname] = value | ||||||
|                 except Exception as e: |                 except Exception as e: | ||||||
|                     raise base.DeserializationError.WithData( |                     raise base.DeserializationError.WithData( | ||||||
|                         e, d["model"], d.get("pk"), field_value |                         e, obj["model"], obj.get("pk"), field_value | ||||||
|                     ) |                     ) | ||||||
|                 if value == base.DEFER_FIELD: |  | ||||||
|                     deferred_fields[field] = field_value |  | ||||||
|                 else: |  | ||||||
|                     data[field.attname] = value |  | ||||||
|             # Handle all other fields |             # Handle all other fields | ||||||
|             else: |             else: | ||||||
|                 try: |                 try: | ||||||
|                     data[field.name] = field.to_python(field_value) |                     data[field.name] = field.to_python(field_value) | ||||||
|                 except Exception as e: |                 except Exception as e: | ||||||
|                     raise base.DeserializationError.WithData( |                     raise base.DeserializationError.WithData( | ||||||
|                         e, d["model"], d.get("pk"), field_value |                         e, obj["model"], obj.get("pk"), field_value | ||||||
|                     ) |                     ) | ||||||
|  |  | ||||||
|         obj = base.build_instance(Model, data, using) |         model_instance = base.build_instance(Model, data, self.using) | ||||||
|         yield base.DeserializedObject(obj, m2m_data, deferred_fields) |         yield base.DeserializedObject(model_instance, m2m_data, deferred_fields) | ||||||
|  |  | ||||||
|  |     def _handle_m2m_field_node(self, field, field_value): | ||||||
| def _get_model(model_identifier): |         return base.deserialize_m2m_values( | ||||||
|     """Look up a model from an "app_label.model_name" string.""" |             field, field_value, self.using, self.handle_forward_references | ||||||
|     try: |  | ||||||
|         return apps.get_model(model_identifier) |  | ||||||
|     except (LookupError, TypeError): |  | ||||||
|         raise base.DeserializationError( |  | ||||||
|             "Invalid model identifier: '%s'" % model_identifier |  | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|  |     def _handle_fk_field_node(self, field, field_value): | ||||||
|  |         return base.deserialize_fk_value( | ||||||
|  |             field, field_value, self.using, self.handle_forward_references | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def _get_model_from_node(model_identifier): | ||||||
|  |         """Look up a model from an "app_label.model_name" string.""" | ||||||
|  |         try: | ||||||
|  |             return apps.get_model(model_identifier) | ||||||
|  |         except (LookupError, TypeError): | ||||||
|  |             raise base.DeserializationError( | ||||||
|  |                 f"Invalid model identifier: {model_identifier}" | ||||||
|  |             ) | ||||||
|   | |||||||
| @@ -6,7 +6,6 @@ Requires PyYaml (https://pyyaml.org/), but that's checked for in __init__. | |||||||
|  |  | ||||||
| import collections | import collections | ||||||
| import decimal | import decimal | ||||||
| from io import StringIO |  | ||||||
|  |  | ||||||
| import yaml | import yaml | ||||||
|  |  | ||||||
| @@ -66,17 +65,23 @@ class Serializer(PythonSerializer): | |||||||
|         return super(PythonSerializer, self).getvalue() |         return super(PythonSerializer, self).getvalue() | ||||||
|  |  | ||||||
|  |  | ||||||
| def Deserializer(stream_or_string, **options): | class Deserializer(PythonDeserializer): | ||||||
|     """Deserialize a stream or string of YAML data.""" |     """Deserialize a stream or string of YAML data.""" | ||||||
|     if isinstance(stream_or_string, bytes): |  | ||||||
|         stream_or_string = stream_or_string.decode() |     def __init__(self, stream_or_string, **options): | ||||||
|     if isinstance(stream_or_string, str): |  | ||||||
|         stream = StringIO(stream_or_string) |  | ||||||
|     else: |  | ||||||
|         stream = stream_or_string |         stream = stream_or_string | ||||||
|     try: |         if isinstance(stream_or_string, bytes): | ||||||
|         yield from PythonDeserializer(yaml.load(stream, Loader=SafeLoader), **options) |             stream = stream_or_string.decode() | ||||||
|     except (GeneratorExit, DeserializationError): |         try: | ||||||
|         raise |             objects = yaml.load(stream, Loader=SafeLoader) | ||||||
|     except Exception as exc: |         except Exception as exc: | ||||||
|         raise DeserializationError() from exc |             raise DeserializationError() from exc | ||||||
|  |         super().__init__(objects, **options) | ||||||
|  |  | ||||||
|  |     def _handle_object(self, obj): | ||||||
|  |         try: | ||||||
|  |             yield from super()._handle_object(obj) | ||||||
|  |         except (GeneratorExit, DeserializationError): | ||||||
|  |             raise | ||||||
|  |         except Exception as exc: | ||||||
|  |             raise DeserializationError(f"Error deserializing object: {exc}") from exc | ||||||
|   | |||||||
| @@ -241,7 +241,9 @@ Security | |||||||
| Serialization | Serialization | ||||||
| ~~~~~~~~~~~~~ | ~~~~~~~~~~~~~ | ||||||
|  |  | ||||||
| * ... | * Each serialization format now defines a ``Deserializer`` class, rather than a | ||||||
|  |   function, to improve extensibility when defining a | ||||||
|  |   :ref:`custom serialization format <custom-serialization-formats>`. | ||||||
|  |  | ||||||
| Signals | Signals | ||||||
| ~~~~~~~ | ~~~~~~~ | ||||||
|   | |||||||
| @@ -347,6 +347,86 @@ again a mapping with the key being name of the field and the value the value: | |||||||
|  |  | ||||||
| Referential fields are again represented by the PK or sequence of PKs. | Referential fields are again represented by the PK or sequence of PKs. | ||||||
|  |  | ||||||
|  | .. _custom-serialization-formats: | ||||||
|  |  | ||||||
|  | Custom serialization formats | ||||||
|  | ---------------------------- | ||||||
|  |  | ||||||
|  | In addition to the default formats, you can create a custom serialization | ||||||
|  | format. | ||||||
|  |  | ||||||
|  | For example, let’s consider a csv serializer and deserializer. First, define a | ||||||
|  | ``Serializer`` and a ``Deserializer`` class. These can override existing | ||||||
|  | serialization format classes: | ||||||
|  |  | ||||||
|  | .. code-block:: python | ||||||
|  |    :caption: ``path/to/custom_csv_serializer.py`` | ||||||
|  |  | ||||||
|  |     import csv | ||||||
|  |  | ||||||
|  |     from django.apps import apps | ||||||
|  |     from django.core import serializers | ||||||
|  |     from django.core.serializers.base import DeserializationError | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     class Serializer(serializers.python.Serializer): | ||||||
|  |         def get_dump_object(self, obj): | ||||||
|  |             dumped_object = super().get_dump_object(obj) | ||||||
|  |             row = [dumped_object["model"], str(dumped_object["pk"])] | ||||||
|  |             row += [str(value) for value in dumped_object["fields"].values()] | ||||||
|  |             return ",".join(row), dumped_object["model"] | ||||||
|  |  | ||||||
|  |         def end_object(self, obj): | ||||||
|  |             dumped_object_str, model = self.get_dump_object(obj) | ||||||
|  |             if self.first: | ||||||
|  |                 fields = [field.name for field in apps.get_model(model)._meta.fields] | ||||||
|  |                 header = ",".join(fields) | ||||||
|  |                 self.stream.write(f"model,{header}\n") | ||||||
|  |             self.stream.write(f"{dumped_object_str}\n") | ||||||
|  |  | ||||||
|  |         def getvalue(self): | ||||||
|  |             return super(serializers.python.Serializer, self).getvalue() | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     class Deserializer(serializers.python.Deserializer): | ||||||
|  |         def __init__(self, stream_or_string, **options): | ||||||
|  |             if isinstance(stream_or_string, bytes): | ||||||
|  |                 stream_or_string = stream_or_string.decode() | ||||||
|  |             if isinstance(stream_or_string, str): | ||||||
|  |                 stream_or_string = stream_or_string.splitlines() | ||||||
|  |             try: | ||||||
|  |                 objects = csv.DictReader(stream_or_string) | ||||||
|  |             except Exception as exc: | ||||||
|  |                 raise DeserializationError() from exc | ||||||
|  |             super().__init__(objects, **options) | ||||||
|  |  | ||||||
|  |         def _handle_object(self, obj): | ||||||
|  |             try: | ||||||
|  |                 model_fields = apps.get_model(obj["model"])._meta.fields | ||||||
|  |                 obj["fields"] = { | ||||||
|  |                     field.name: obj[field.name] | ||||||
|  |                     for field in model_fields | ||||||
|  |                     if field.name in obj | ||||||
|  |                 } | ||||||
|  |                 yield from super()._handle_object(obj) | ||||||
|  |             except (GeneratorExit, DeserializationError): | ||||||
|  |                 raise | ||||||
|  |             except Exception as exc: | ||||||
|  |                 raise DeserializationError(f"Error deserializing object: {exc}") from exc | ||||||
|  |  | ||||||
|  | Then add the module containing the serializer definitions to your | ||||||
|  | :setting:`SERIALIZATION_MODULES` setting:: | ||||||
|  |  | ||||||
|  |     SERIALIZATION_MODULES = { | ||||||
|  |         "csv": "path.to.custom_csv_serializer", | ||||||
|  |         "json": "django.core.serializers.json", | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | .. versionchanged:: 5.2 | ||||||
|  |  | ||||||
|  |     A ``Deserializer`` class definition was added to each of the provided | ||||||
|  |     serialization formats. | ||||||
|  |  | ||||||
| .. _topics-serialization-natural-keys: | .. _topics-serialization-natural-keys: | ||||||
|  |  | ||||||
| Natural keys | Natural keys | ||||||
|   | |||||||
							
								
								
									
										125
									
								
								tests/serializers/test_deserialization.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										125
									
								
								tests/serializers/test_deserialization.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,125 @@ | |||||||
|  | import json | ||||||
|  |  | ||||||
|  | from django.core.serializers.base import DeserializationError, DeserializedObject | ||||||
|  | from django.core.serializers.json import Deserializer as JsonDeserializer | ||||||
|  | from django.core.serializers.jsonl import Deserializer as JsonlDeserializer | ||||||
|  | from django.core.serializers.python import Deserializer | ||||||
|  | from django.core.serializers.pyyaml import Deserializer as YamlDeserializer | ||||||
|  | from django.test import SimpleTestCase | ||||||
|  |  | ||||||
|  | from .models import Author | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TestDeserializer(SimpleTestCase): | ||||||
|  |     def setUp(self): | ||||||
|  |         self.object_list = [ | ||||||
|  |             {"pk": 1, "model": "serializers.author", "fields": {"name": "Jane"}}, | ||||||
|  |             {"pk": 2, "model": "serializers.author", "fields": {"name": "Joe"}}, | ||||||
|  |         ] | ||||||
|  |         self.deserializer = Deserializer(self.object_list) | ||||||
|  |         self.jane = Author(name="Jane", pk=1) | ||||||
|  |         self.joe = Author(name="Joe", pk=2) | ||||||
|  |  | ||||||
|  |     def test_deserialized_object_repr(self): | ||||||
|  |         deserial_obj = DeserializedObject(obj=self.jane) | ||||||
|  |         self.assertEqual( | ||||||
|  |             repr(deserial_obj), "<DeserializedObject: serializers.Author(pk=1)>" | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     def test_next_functionality(self): | ||||||
|  |         first_item = next(self.deserializer) | ||||||
|  |  | ||||||
|  |         self.assertEqual(first_item.object, self.jane) | ||||||
|  |  | ||||||
|  |         second_item = next(self.deserializer) | ||||||
|  |         self.assertEqual(second_item.object, self.joe) | ||||||
|  |  | ||||||
|  |         with self.assertRaises(StopIteration): | ||||||
|  |             next(self.deserializer) | ||||||
|  |  | ||||||
|  |     def test_invalid_model_identifier(self): | ||||||
|  |         invalid_object_list = [ | ||||||
|  |             {"pk": 1, "model": "serializers.author2", "fields": {"name": "Jane"}} | ||||||
|  |         ] | ||||||
|  |         self.deserializer = Deserializer(invalid_object_list) | ||||||
|  |         with self.assertRaises(DeserializationError): | ||||||
|  |             next(self.deserializer) | ||||||
|  |  | ||||||
|  |         deserializer = Deserializer(object_list=[]) | ||||||
|  |         with self.assertRaises(StopIteration): | ||||||
|  |             next(deserializer) | ||||||
|  |  | ||||||
|  |     def test_custom_deserializer(self): | ||||||
|  |         class CustomDeserializer(Deserializer): | ||||||
|  |             @staticmethod | ||||||
|  |             def _get_model_from_node(model_identifier): | ||||||
|  |                 return Author | ||||||
|  |  | ||||||
|  |         deserializer = CustomDeserializer(self.object_list) | ||||||
|  |         result = next(iter(deserializer)) | ||||||
|  |         deserialized_object = result.object | ||||||
|  |         self.assertEqual( | ||||||
|  |             self.jane, | ||||||
|  |             deserialized_object, | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     def test_empty_object_list(self): | ||||||
|  |         deserializer = Deserializer(object_list=[]) | ||||||
|  |         with self.assertRaises(StopIteration): | ||||||
|  |             next(deserializer) | ||||||
|  |  | ||||||
|  |     def test_json_bytes_input(self): | ||||||
|  |         test_string = json.dumps(self.object_list) | ||||||
|  |         stream = test_string.encode("utf-8") | ||||||
|  |         deserializer = JsonDeserializer(stream_or_string=stream) | ||||||
|  |  | ||||||
|  |         first_item = next(deserializer) | ||||||
|  |         second_item = next(deserializer) | ||||||
|  |  | ||||||
|  |         self.assertEqual(first_item.object, self.jane) | ||||||
|  |         self.assertEqual(second_item.object, self.joe) | ||||||
|  |  | ||||||
|  |     def test_jsonl_bytes_input(self): | ||||||
|  |         test_string = """ | ||||||
|  |         {"pk": 1, "model": "serializers.author", "fields": {"name": "Jane"}} | ||||||
|  |         {"pk": 2, "model": "serializers.author", "fields": {"name": "Joe"}} | ||||||
|  |         {"pk": 3, "model": "serializers.author", "fields": {"name": "John"}} | ||||||
|  |         {"pk": 4, "model": "serializers.author", "fields": {"name": "Smith"}}""" | ||||||
|  |         stream = test_string.encode("utf-8") | ||||||
|  |         deserializer = JsonlDeserializer(stream_or_string=stream) | ||||||
|  |  | ||||||
|  |         first_item = next(deserializer) | ||||||
|  |         second_item = next(deserializer) | ||||||
|  |  | ||||||
|  |         self.assertEqual(first_item.object, self.jane) | ||||||
|  |         self.assertEqual(second_item.object, self.joe) | ||||||
|  |  | ||||||
|  |     def test_yaml_bytes_input(self): | ||||||
|  |         test_string = """- pk: 1 | ||||||
|  |   model: serializers.author | ||||||
|  |   fields: | ||||||
|  |     name: Jane | ||||||
|  |  | ||||||
|  | - pk: 2 | ||||||
|  |   model: serializers.author | ||||||
|  |   fields: | ||||||
|  |     name: Joe | ||||||
|  |  | ||||||
|  | - pk: 3 | ||||||
|  |   model: serializers.author | ||||||
|  |   fields: | ||||||
|  |     name: John | ||||||
|  |  | ||||||
|  | - pk: 4 | ||||||
|  |   model: serializers.author | ||||||
|  |   fields: | ||||||
|  |     name: Smith | ||||||
|  | """ | ||||||
|  |         stream = test_string.encode("utf-8") | ||||||
|  |         deserializer = YamlDeserializer(stream_or_string=stream) | ||||||
|  |  | ||||||
|  |         first_item = next(deserializer) | ||||||
|  |         second_item = next(deserializer) | ||||||
|  |  | ||||||
|  |         self.assertEqual(first_item.object, self.jane) | ||||||
|  |         self.assertEqual(second_item.object, self.joe) | ||||||
| @@ -1,13 +0,0 @@ | |||||||
| from django.core.serializers.base import DeserializedObject |  | ||||||
| from django.test import SimpleTestCase |  | ||||||
|  |  | ||||||
| from .models import Author |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class TestDeserializedObjectTests(SimpleTestCase): |  | ||||||
|     def test_repr(self): |  | ||||||
|         author = Author(name="John", pk=1) |  | ||||||
|         deserial_obj = DeserializedObject(obj=author) |  | ||||||
|         self.assertEqual( |  | ||||||
|             repr(deserial_obj), "<DeserializedObject: serializers.Author(pk=1)>" |  | ||||||
|         ) |  | ||||||
		Reference in New Issue
	
	Block a user