2006-06-29 00:00:37 +08:00
|
|
|
"""
|
|
|
|
Module for abstract serializer/unserializer base classes.
|
|
|
|
"""
|
2021-09-02 17:15:40 +08:00
|
|
|
import pickle
|
2020-01-30 17:28:32 +08:00
|
|
|
import warnings
|
2017-01-07 19:11:46 +08:00
|
|
|
from io import StringIO
|
|
|
|
|
2018-07-14 05:54:47 +08:00
|
|
|
from django.core.exceptions import ObjectDoesNotExist
|
2006-06-29 00:00:37 +08:00
|
|
|
from django.db import models
|
2020-01-30 17:28:32 +08:00
|
|
|
from django.utils.deprecation import RemovedInDjango50Warning
|
2006-06-29 00:00:37 +08:00
|
|
|
|
2018-07-14 05:54:47 +08:00
|
|
|
DEFER_FIELD = object()
|
|
|
|
|
2013-11-03 04:12:09 +08:00
|
|
|
|
2021-09-02 17:15:40 +08:00
|
|
|
class PickleSerializer:
|
|
|
|
"""
|
|
|
|
Simple wrapper around pickle to be used in signing.dumps()/loads() and
|
|
|
|
cache backends.
|
|
|
|
"""
|
2022-02-04 03:24:19 +08:00
|
|
|
|
2021-09-02 17:15:40 +08:00
|
|
|
def __init__(self, protocol=None):
|
2020-01-30 17:28:32 +08:00
|
|
|
warnings.warn(
|
|
|
|
"PickleSerializer is deprecated due to its security risk. Use "
|
|
|
|
"JSONSerializer instead.",
|
|
|
|
RemovedInDjango50Warning,
|
|
|
|
)
|
2021-09-02 17:15:40 +08:00
|
|
|
self.protocol = pickle.HIGHEST_PROTOCOL if protocol is None else protocol
|
|
|
|
|
|
|
|
def dumps(self, obj):
|
|
|
|
return pickle.dumps(obj, self.protocol)
|
|
|
|
|
|
|
|
def loads(self, data):
|
|
|
|
return pickle.loads(data)
|
|
|
|
|
|
|
|
|
2011-04-27 00:49:32 +08:00
|
|
|
class SerializerDoesNotExist(KeyError):
|
|
|
|
"""The requested serializer was not found."""
|
2022-02-04 03:24:19 +08:00
|
|
|
|
2011-04-27 00:49:32 +08:00
|
|
|
pass
|
|
|
|
|
2013-11-03 04:12:09 +08:00
|
|
|
|
2006-06-29 00:00:37 +08:00
|
|
|
class SerializationError(Exception):
|
|
|
|
"""Something bad happened during serialization."""
|
2022-02-04 03:24:19 +08:00
|
|
|
|
2006-06-29 00:00:37 +08:00
|
|
|
pass
|
2006-08-31 12:11:46 +08:00
|
|
|
|
2013-11-03 04:12:09 +08:00
|
|
|
|
2006-06-29 00:00:37 +08:00
|
|
|
class DeserializationError(Exception):
|
|
|
|
"""Something bad happened during deserialization."""
|
2015-03-27 02:31:09 +08:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def WithData(cls, original_exc, model, fk, field_value):
|
|
|
|
"""
|
|
|
|
Factory method for creating a deserialization error which has a more
|
2016-05-02 23:43:03 +08:00
|
|
|
explanatory message.
|
2015-03-27 02:31:09 +08:00
|
|
|
"""
|
|
|
|
return cls(
|
|
|
|
"%s: (%s:pk=%s) field_value was '%s'"
|
|
|
|
% (original_exc, model, fk, field_value)
|
2022-02-04 03:24:19 +08:00
|
|
|
)
|
2006-06-29 00:00:37 +08:00
|
|
|
|
2013-11-03 04:12:09 +08:00
|
|
|
|
2018-03-21 00:07:39 +08:00
|
|
|
class M2MDeserializationError(Exception):
|
|
|
|
"""Something bad happened during deserialization of a ManyToManyField."""
|
2022-02-04 03:24:19 +08:00
|
|
|
|
2018-03-21 00:07:39 +08:00
|
|
|
def __init__(self, original_exc, pk):
|
|
|
|
self.original_exc = original_exc
|
|
|
|
self.pk = pk
|
|
|
|
|
|
|
|
|
2017-01-19 15:39:46 +08:00
|
|
|
class ProgressBar:
|
2015-07-22 05:24:32 +08:00
|
|
|
progress_width = 75
|
|
|
|
|
|
|
|
def __init__(self, output, total_count):
|
|
|
|
self.output = output
|
|
|
|
self.total_count = total_count
|
|
|
|
self.prev_done = 0
|
|
|
|
|
|
|
|
def update(self, count):
|
|
|
|
if not self.output:
|
|
|
|
return
|
|
|
|
perc = count * 100 // self.total_count
|
|
|
|
done = perc * self.progress_width // 100
|
|
|
|
if self.prev_done >= done:
|
|
|
|
return
|
|
|
|
self.prev_done = done
|
|
|
|
cr = "" if self.total_count == 1 else "\r"
|
|
|
|
self.output.write(
|
|
|
|
cr + "[" + "." * done + " " * (self.progress_width - done) + "]"
|
|
|
|
)
|
|
|
|
if done == self.progress_width:
|
|
|
|
self.output.write("\n")
|
|
|
|
self.output.flush()
|
|
|
|
|
|
|
|
|
2017-01-19 15:39:46 +08:00
|
|
|
class Serializer:
|
2006-06-29 00:00:37 +08:00
|
|
|
"""
|
|
|
|
Abstract serializer base class.
|
|
|
|
"""
|
2006-08-31 12:11:46 +08:00
|
|
|
|
2008-03-18 22:35:05 +08:00
|
|
|
# Indicates if the implemented serializer is only available for
|
2007-12-17 14:53:15 +08:00
|
|
|
# internal Django use.
|
|
|
|
internal_use_only = False
|
2015-07-22 05:24:32 +08:00
|
|
|
progress_class = ProgressBar
|
2017-01-07 19:11:46 +08:00
|
|
|
stream_class = StringIO
|
2008-03-18 22:35:05 +08:00
|
|
|
|
2017-02-02 00:41:56 +08:00
|
|
|
def serialize(
|
|
|
|
self,
|
|
|
|
queryset,
|
|
|
|
*,
|
|
|
|
stream=None,
|
|
|
|
fields=None,
|
|
|
|
use_natural_foreign_keys=False,
|
|
|
|
use_natural_primary_keys=False,
|
|
|
|
progress_output=None,
|
|
|
|
object_count=0,
|
|
|
|
**options,
|
|
|
|
):
|
2006-06-29 00:00:37 +08:00
|
|
|
"""
|
|
|
|
Serialize a queryset.
|
|
|
|
"""
|
|
|
|
self.options = options
|
2006-08-31 12:11:46 +08:00
|
|
|
|
2017-02-02 00:41:56 +08:00
|
|
|
self.stream = stream if stream is not None else self.stream_class()
|
|
|
|
self.selected_fields = fields
|
|
|
|
self.use_natural_foreign_keys = use_natural_foreign_keys
|
|
|
|
self.use_natural_primary_keys = use_natural_primary_keys
|
|
|
|
progress_bar = self.progress_class(progress_output, object_count)
|
2006-08-31 12:11:46 +08:00
|
|
|
|
2006-06-29 00:00:37 +08:00
|
|
|
self.start_serialization()
|
2012-05-26 17:43:37 +08:00
|
|
|
self.first = True
|
2015-07-22 05:24:32 +08:00
|
|
|
for count, obj in enumerate(queryset, start=1):
|
2006-06-29 00:00:37 +08:00
|
|
|
self.start_object(obj)
|
2012-03-04 01:50:18 +08:00
|
|
|
# Use the concrete parent class' _meta instead of the object's _meta
|
|
|
|
# This is to avoid local_fields problems for proxy models. Refs #17717.
|
2012-03-04 05:34:51 +08:00
|
|
|
concrete_model = obj._meta.concrete_model
|
2017-02-11 19:40:15 +08:00
|
|
|
# When using natural primary keys, retrieve the pk field of the
|
|
|
|
# parent for multi-table inheritance child models. That field must
|
|
|
|
# be serialized, otherwise deserialization isn't possible.
|
|
|
|
if self.use_natural_primary_keys:
|
|
|
|
pk = concrete_model._meta.pk
|
|
|
|
pk_parent = (
|
|
|
|
pk if pk.remote_field and pk.remote_field.parent_link else None
|
2022-02-04 03:24:19 +08:00
|
|
|
)
|
2017-02-11 19:40:15 +08:00
|
|
|
else:
|
|
|
|
pk_parent = None
|
2012-03-04 05:34:51 +08:00
|
|
|
for field in concrete_model._meta.local_fields:
|
2017-02-11 19:40:15 +08:00
|
|
|
if field.serialize or field is pk_parent:
|
2015-02-26 22:19:17 +08:00
|
|
|
if field.remote_field is None:
|
2007-03-19 19:57:53 +08:00
|
|
|
if (
|
|
|
|
self.selected_fields is None
|
|
|
|
or field.attname in self.selected_fields
|
|
|
|
):
|
|
|
|
self.handle_field(obj, field)
|
|
|
|
else:
|
2017-02-11 19:31:20 +08:00
|
|
|
if (
|
|
|
|
self.selected_fields is None
|
|
|
|
or field.attname[:-3] in self.selected_fields
|
|
|
|
):
|
2007-03-19 19:57:53 +08:00
|
|
|
self.handle_fk_field(obj, field)
|
2019-06-27 15:47:13 +08:00
|
|
|
for field in concrete_model._meta.local_many_to_many:
|
2007-03-19 19:57:53 +08:00
|
|
|
if field.serialize:
|
|
|
|
if (
|
|
|
|
self.selected_fields is None
|
|
|
|
or field.attname in self.selected_fields
|
|
|
|
):
|
|
|
|
self.handle_m2m_field(obj, field)
|
2006-06-29 00:00:37 +08:00
|
|
|
self.end_object(obj)
|
2015-07-22 05:24:32 +08:00
|
|
|
progress_bar.update(count)
|
2018-01-04 07:52:12 +08:00
|
|
|
self.first = self.first and False
|
2006-06-29 00:00:37 +08:00
|
|
|
self.end_serialization()
|
|
|
|
return self.getvalue()
|
2006-08-31 12:11:46 +08:00
|
|
|
|
2006-06-29 00:00:37 +08:00
|
|
|
def start_serialization(self):
|
|
|
|
"""
|
|
|
|
Called when serializing of the queryset starts.
|
|
|
|
"""
|
2013-09-07 02:24:52 +08:00
|
|
|
raise NotImplementedError(
|
|
|
|
"subclasses of Serializer must provide a start_serialization() method"
|
|
|
|
)
|
2006-08-31 12:11:46 +08:00
|
|
|
|
2006-06-29 00:00:37 +08:00
|
|
|
def end_serialization(self):
|
|
|
|
"""
|
|
|
|
Called when serializing of the queryset ends.
|
|
|
|
"""
|
|
|
|
pass
|
2006-08-31 12:11:46 +08:00
|
|
|
|
2006-06-29 00:00:37 +08:00
|
|
|
def start_object(self, obj):
|
|
|
|
"""
|
|
|
|
Called when serializing of an object starts.
|
|
|
|
"""
|
2013-09-07 02:24:52 +08:00
|
|
|
raise NotImplementedError(
|
|
|
|
"subclasses of Serializer must provide a start_object() method"
|
|
|
|
)
|
2006-08-31 12:11:46 +08:00
|
|
|
|
2006-06-29 00:00:37 +08:00
|
|
|
def end_object(self, obj):
|
|
|
|
"""
|
|
|
|
Called when serializing of an object ends.
|
|
|
|
"""
|
|
|
|
pass
|
2006-08-31 12:11:46 +08:00
|
|
|
|
2006-06-29 00:00:37 +08:00
|
|
|
def handle_field(self, obj, field):
|
|
|
|
"""
|
|
|
|
Called to handle each individual (non-relational) field on an object.
|
|
|
|
"""
|
2019-04-14 15:44:56 +08:00
|
|
|
raise NotImplementedError(
|
|
|
|
"subclasses of Serializer must provide a handle_field() method"
|
|
|
|
)
|
2006-08-31 12:11:46 +08:00
|
|
|
|
2006-06-29 00:00:37 +08:00
|
|
|
def handle_fk_field(self, obj, field):
|
|
|
|
"""
|
|
|
|
Called to handle a ForeignKey field.
|
|
|
|
"""
|
2019-04-14 15:44:56 +08:00
|
|
|
raise NotImplementedError(
|
|
|
|
"subclasses of Serializer must provide a handle_fk_field() method"
|
|
|
|
)
|
2006-08-31 12:11:46 +08:00
|
|
|
|
2006-06-29 00:00:37 +08:00
|
|
|
def handle_m2m_field(self, obj, field):
|
|
|
|
"""
|
|
|
|
Called to handle a ManyToManyField.
|
|
|
|
"""
|
2019-04-14 15:44:56 +08:00
|
|
|
raise NotImplementedError(
|
|
|
|
"subclasses of Serializer must provide a handle_m2m_field() method"
|
|
|
|
)
|
2006-08-31 12:11:46 +08:00
|
|
|
|
2006-06-29 00:00:37 +08:00
|
|
|
def getvalue(self):
|
|
|
|
"""
|
2007-04-25 18:12:05 +08:00
|
|
|
Return the fully serialized queryset (or None if the output stream is
|
|
|
|
not seekable).
|
2006-06-29 00:00:37 +08:00
|
|
|
"""
|
2007-04-25 18:12:05 +08:00
|
|
|
if callable(getattr(self.stream, "getvalue", None)):
|
|
|
|
return self.stream.getvalue()
|
2006-06-29 00:00:37 +08:00
|
|
|
|
2013-11-03 04:12:09 +08:00
|
|
|
|
2017-01-07 19:11:46 +08:00
|
|
|
class Deserializer:
|
2006-06-29 00:00:37 +08:00
|
|
|
"""
|
|
|
|
Abstract base deserializer class.
|
|
|
|
"""
|
2006-08-31 12:11:46 +08:00
|
|
|
|
2006-06-29 00:00:37 +08:00
|
|
|
def __init__(self, stream_or_string, **options):
|
|
|
|
"""
|
|
|
|
Init this serializer given a stream or a string
|
|
|
|
"""
|
|
|
|
self.options = options
|
2016-12-29 23:27:49 +08:00
|
|
|
if isinstance(stream_or_string, str):
|
2017-01-07 19:11:46 +08:00
|
|
|
self.stream = StringIO(stream_or_string)
|
2006-06-29 00:00:37 +08:00
|
|
|
else:
|
|
|
|
self.stream = stream_or_string
|
2006-08-31 12:11:46 +08:00
|
|
|
|
2006-06-29 00:00:37 +08:00
|
|
|
def __iter__(self):
|
|
|
|
return self
|
2006-08-31 12:11:46 +08:00
|
|
|
|
2012-08-09 20:36:05 +08:00
|
|
|
def __next__(self):
|
2018-08-02 00:55:53 +08:00
|
|
|
"""Iteration interface -- return the next item in the stream"""
|
2013-09-07 02:24:52 +08:00
|
|
|
raise NotImplementedError(
|
|
|
|
"subclasses of Deserializer must provide a __next__() method"
|
|
|
|
)
|
2006-08-31 12:11:46 +08:00
|
|
|
|
2013-11-03 04:12:09 +08:00
|
|
|
|
2017-01-19 15:39:46 +08:00
|
|
|
class DeserializedObject:
|
2006-06-29 00:00:37 +08:00
|
|
|
"""
|
2007-03-01 21:11:08 +08:00
|
|
|
A deserialized model.
|
2006-08-31 12:11:46 +08:00
|
|
|
|
2006-06-29 00:00:37 +08:00
|
|
|
Basically a container for holding the pre-saved deserialized data along
|
|
|
|
with the many-to-many data saved with the object.
|
2006-08-31 12:11:46 +08:00
|
|
|
|
2006-06-29 00:00:37 +08:00
|
|
|
Call ``save()`` to save the object (with the many-to-many data) to the
|
|
|
|
database; call ``save(save_m2m=False)`` to save just the object fields
|
|
|
|
(and not touch the many-to-many stuff.)
|
|
|
|
"""
|
2006-08-31 12:11:46 +08:00
|
|
|
|
2018-07-14 05:54:47 +08:00
|
|
|
def __init__(self, obj, m2m_data=None, deferred_fields=None):
|
2006-06-29 00:00:37 +08:00
|
|
|
self.object = obj
|
|
|
|
self.m2m_data = m2m_data
|
2018-07-14 05:54:47 +08:00
|
|
|
self.deferred_fields = deferred_fields
|
2006-08-31 12:11:46 +08:00
|
|
|
|
2006-06-29 00:00:37 +08:00
|
|
|
def __repr__(self):
|
2016-12-01 21:08:51 +08:00
|
|
|
return "<%s: %s(pk=%s)>" % (
|
|
|
|
self.__class__.__name__,
|
|
|
|
self.object._meta.label,
|
|
|
|
self.object.pk,
|
|
|
|
)
|
2006-08-31 12:11:46 +08:00
|
|
|
|
2015-06-05 22:58:36 +08:00
|
|
|
def save(self, save_m2m=True, using=None, **kwargs):
|
2008-03-18 22:35:05 +08:00
|
|
|
# Call save on the Model baseclass directly. This bypasses any
|
2007-07-12 15:45:35 +08:00
|
|
|
# model-defined save. The save is also forced to be raw.
|
2013-05-12 07:34:02 +08:00
|
|
|
# raw=True is passed to any pre/post_save signals.
|
2015-06-05 22:58:36 +08:00
|
|
|
models.Model.save_base(self.object, using=using, raw=True, **kwargs)
|
2006-06-29 00:00:37 +08:00
|
|
|
if self.m2m_data and save_m2m:
|
|
|
|
for accessor_name, object_list in self.m2m_data.items():
|
2015-10-09 05:17:10 +08:00
|
|
|
getattr(self.object, accessor_name).set(object_list)
|
2006-08-31 12:11:46 +08:00
|
|
|
|
|
|
|
# prevent a second (possibly accidental) call to save() from saving
|
2006-06-29 00:00:37 +08:00
|
|
|
# the m2m data twice.
|
|
|
|
self.m2m_data = None
|
2012-08-01 09:49:01 +08:00
|
|
|
|
2018-07-14 05:54:47 +08:00
|
|
|
def save_deferred_fields(self, using=None):
|
|
|
|
self.m2m_data = {}
|
|
|
|
for field, field_value in self.deferred_fields.items():
|
|
|
|
opts = self.object._meta
|
|
|
|
label = opts.app_label + "." + opts.model_name
|
|
|
|
if isinstance(field.remote_field, models.ManyToManyRel):
|
|
|
|
try:
|
|
|
|
values = deserialize_m2m_values(
|
|
|
|
field, field_value, using, handle_forward_references=False
|
|
|
|
)
|
|
|
|
except M2MDeserializationError as e:
|
|
|
|
raise DeserializationError.WithData(
|
|
|
|
e.original_exc, label, self.object.pk, e.pk
|
|
|
|
)
|
|
|
|
self.m2m_data[field.name] = values
|
|
|
|
elif isinstance(field.remote_field, models.ManyToOneRel):
|
|
|
|
try:
|
|
|
|
value = deserialize_fk_value(
|
|
|
|
field, field_value, using, handle_forward_references=False
|
|
|
|
)
|
|
|
|
except Exception as e:
|
|
|
|
raise DeserializationError.WithData(
|
|
|
|
e, label, self.object.pk, field_value
|
|
|
|
)
|
|
|
|
setattr(self.object, field.attname, value)
|
|
|
|
self.save()
|
|
|
|
|
2013-11-03 04:12:09 +08:00
|
|
|
|
2012-08-01 09:49:01 +08:00
|
|
|
def build_instance(Model, data, db):
|
|
|
|
"""
|
|
|
|
Build a model instance.
|
|
|
|
|
|
|
|
If the model instance doesn't have a primary key and the model supports
|
|
|
|
natural keys, try to retrieve it from the database.
|
|
|
|
"""
|
2018-11-23 05:11:51 +08:00
|
|
|
default_manager = Model._meta.default_manager
|
2021-02-05 01:19:14 +08:00
|
|
|
pk = data.get(Model._meta.pk.attname)
|
2018-11-23 05:11:51 +08:00
|
|
|
if (
|
|
|
|
pk is None
|
|
|
|
and hasattr(default_manager, "get_by_natural_key")
|
|
|
|
and hasattr(Model, "natural_key")
|
|
|
|
):
|
2022-03-18 18:17:54 +08:00
|
|
|
obj = Model(**data)
|
|
|
|
obj._state.db = db
|
|
|
|
natural_key = obj.natural_key()
|
2012-08-01 09:49:01 +08:00
|
|
|
try:
|
2018-11-23 05:11:51 +08:00
|
|
|
data[Model._meta.pk.attname] = Model._meta.pk.to_python(
|
|
|
|
default_manager.db_manager(db).get_by_natural_key(*natural_key).pk
|
|
|
|
)
|
2012-08-01 09:49:01 +08:00
|
|
|
except Model.DoesNotExist:
|
|
|
|
pass
|
2018-11-23 05:11:51 +08:00
|
|
|
return Model(**data)
|
2018-03-21 00:07:39 +08:00
|
|
|
|
|
|
|
|
2018-07-14 05:54:47 +08:00
|
|
|
def deserialize_m2m_values(field, field_value, using, handle_forward_references):
|
2018-03-21 00:07:39 +08:00
|
|
|
model = field.remote_field.model
|
|
|
|
if hasattr(model._default_manager, "get_by_natural_key"):
|
2022-02-04 03:24:19 +08:00
|
|
|
|
2018-03-21 00:07:39 +08:00
|
|
|
def m2m_convert(value):
|
|
|
|
if hasattr(value, "__iter__") and not isinstance(value, str):
|
|
|
|
return (
|
|
|
|
model._default_manager.db_manager(using)
|
|
|
|
.get_by_natural_key(*value)
|
|
|
|
.pk
|
2022-02-04 03:24:19 +08:00
|
|
|
)
|
2018-03-21 00:07:39 +08:00
|
|
|
else:
|
|
|
|
return model._meta.pk.to_python(value)
|
2022-02-04 03:24:19 +08:00
|
|
|
|
2018-03-21 00:07:39 +08:00
|
|
|
else:
|
2022-02-04 03:24:19 +08:00
|
|
|
|
2018-03-21 00:07:39 +08:00
|
|
|
def m2m_convert(v):
|
|
|
|
return model._meta.pk.to_python(v)
|
|
|
|
|
2019-12-12 02:36:57 +08:00
|
|
|
try:
|
|
|
|
pks_iter = iter(field_value)
|
|
|
|
except TypeError as e:
|
|
|
|
raise M2MDeserializationError(e, field_value)
|
2018-03-21 00:07:39 +08:00
|
|
|
try:
|
|
|
|
values = []
|
2019-12-12 02:36:57 +08:00
|
|
|
for pk in pks_iter:
|
2018-03-21 00:07:39 +08:00
|
|
|
values.append(m2m_convert(pk))
|
|
|
|
return values
|
|
|
|
except Exception as e:
|
2018-07-14 05:54:47 +08:00
|
|
|
if isinstance(e, ObjectDoesNotExist) and handle_forward_references:
|
|
|
|
return DEFER_FIELD
|
|
|
|
else:
|
|
|
|
raise M2MDeserializationError(e, pk)
|
2018-03-21 00:07:39 +08:00
|
|
|
|
|
|
|
|
2018-07-14 05:54:47 +08:00
|
|
|
def deserialize_fk_value(field, field_value, using, handle_forward_references):
|
2018-03-21 00:07:39 +08:00
|
|
|
if field_value is None:
|
|
|
|
return None
|
|
|
|
model = field.remote_field.model
|
|
|
|
default_manager = model._default_manager
|
|
|
|
field_name = field.remote_field.field_name
|
|
|
|
if (
|
|
|
|
hasattr(default_manager, "get_by_natural_key")
|
|
|
|
and hasattr(field_value, "__iter__")
|
|
|
|
and not isinstance(field_value, str)
|
|
|
|
):
|
2018-07-14 05:54:47 +08:00
|
|
|
try:
|
|
|
|
obj = default_manager.db_manager(using).get_by_natural_key(*field_value)
|
|
|
|
except ObjectDoesNotExist:
|
|
|
|
if handle_forward_references:
|
|
|
|
return DEFER_FIELD
|
|
|
|
else:
|
|
|
|
raise
|
2018-03-21 00:07:39 +08:00
|
|
|
value = getattr(obj, field_name)
|
|
|
|
# If this is a natural foreign key to an object that has a FK/O2O as
|
|
|
|
# the foreign key, use the FK value.
|
|
|
|
if model._meta.pk.remote_field:
|
|
|
|
value = value.pk
|
|
|
|
return value
|
|
|
|
return model._meta.get_field(field_name).to_python(field_value)
|