mirror of
https://github.com/encode/django-rest-framework.git
synced 2025-10-24 04:31:08 +03:00
Add flag in get_related_field
This commit is contained in:
parent
b0c370dd2b
commit
9bbc1cc403
|
@ -120,7 +120,7 @@ Let's look at an example of serializing a class that represents an RGB color val
|
||||||
assert(red < 256 and green < 256 and blue < 256)
|
assert(red < 256 and green < 256 and blue < 256)
|
||||||
self.red, self.green, self.blue = red, green, blue
|
self.red, self.green, self.blue = red, green, blue
|
||||||
|
|
||||||
class ColourField(Field):
|
class ColourField(serializers.WritableField):
|
||||||
"""
|
"""
|
||||||
Color objects are serialized into "rgb(#, #, #)" notation.
|
Color objects are serialized into "rgb(#, #, #)" notation.
|
||||||
"""
|
"""
|
||||||
|
@ -138,7 +138,7 @@ By default field values are treated as mapping to an attribute on the object. I
|
||||||
|
|
||||||
As an example, let's create a field that can be used represent the class name of the object being serialized:
|
As an example, let's create a field that can be used represent the class name of the object being serialized:
|
||||||
|
|
||||||
class ClassNameField(Field):
|
class ClassNameField(serializers.WritableField):
|
||||||
def field_to_native(self, obj, field_name):
|
def field_to_native(self, obj, field_name):
|
||||||
"""
|
"""
|
||||||
Serialize the object's class name, not an attribute of the object.
|
Serialize the object's class name, not an attribute of the object.
|
||||||
|
@ -158,7 +158,7 @@ As an example, let's create a field that can be used represent the class name of
|
||||||
Often you'll want serializer classes that map closely to model definitions.
|
Often you'll want serializer classes that map closely to model definitions.
|
||||||
The `ModelSerializer` class lets you automatically create a Serializer class with fields that corrospond to the Model fields.
|
The `ModelSerializer` class lets you automatically create a Serializer class with fields that corrospond to the Model fields.
|
||||||
|
|
||||||
class AccountSerializer(ModelSerializer):
|
class AccountSerializer(serializers.ModelSerializer):
|
||||||
class Meta:
|
class Meta:
|
||||||
model = Account
|
model = Account
|
||||||
|
|
||||||
|
@ -168,7 +168,7 @@ The `ModelSerializer` class lets you automatically create a Serializer class wit
|
||||||
|
|
||||||
You can add extra fields to a `ModelSerializer` or override the default fields by declaring fields on the class, just as you would for a `Serializer` class.
|
You can add extra fields to a `ModelSerializer` or override the default fields by declaring fields on the class, just as you would for a `Serializer` class.
|
||||||
|
|
||||||
class AccountSerializer(ModelSerializer):
|
class AccountSerializer(serializers.ModelSerializer):
|
||||||
url = CharField(source='get_absolute_url', readonly=True)
|
url = CharField(source='get_absolute_url', readonly=True)
|
||||||
group = NaturalKeyField()
|
group = NaturalKeyField()
|
||||||
|
|
||||||
|
@ -183,7 +183,7 @@ When serializing model instances, there are a number of different ways you might
|
||||||
|
|
||||||
Alternative representations include serializing using natural keys, serializing complete nested representations, or serializing using a custom representation, such as a URL that uniquely identifies the model instances.
|
Alternative representations include serializing using natural keys, serializing complete nested representations, or serializing using a custom representation, such as a URL that uniquely identifies the model instances.
|
||||||
|
|
||||||
The `PrimaryKeyField` and `NaturalKeyField` fields provide alternative flat representations.
|
The `PrimaryKeyRelatedField` and `HyperlinkedRelatedField` fields provide alternative flat representations.
|
||||||
|
|
||||||
The `ModelSerializer` class can itself be used as a field, in order to serialize relationships using nested representations.
|
The `ModelSerializer` class can itself be used as a field, in order to serialize relationships using nested representations.
|
||||||
|
|
||||||
|
@ -197,20 +197,16 @@ If you only want a subset of the default fields to be used in a model serializer
|
||||||
|
|
||||||
For example:
|
For example:
|
||||||
|
|
||||||
class AccountSerializer(ModelSerializer):
|
class AccountSerializer(serializers.ModelSerializer):
|
||||||
class Meta:
|
class Meta:
|
||||||
model = Account
|
model = Account
|
||||||
exclude = ('id',)
|
exclude = ('id',)
|
||||||
|
|
||||||
The `fields` and `exclude` options may also be set by passing them to the `serialize()` method.
|
|
||||||
|
|
||||||
**[TODO: Possibly only allow .serialize(fields=…) in FixtureSerializer for backwards compatability, but remove for ModelSerializer]**
|
|
||||||
|
|
||||||
## Specifiying nested serialization
|
## Specifiying nested serialization
|
||||||
|
|
||||||
The default `ModelSerializer` uses primary keys for relationships, but you can also easily generate nested representations using the `nested` option:
|
The default `ModelSerializer` uses primary keys for relationships, but you can also easily generate nested representations using the `nested` option:
|
||||||
|
|
||||||
class AccountSerializer(ModelSerializer):
|
class AccountSerializer(serializers.ModelSerializer):
|
||||||
class Meta:
|
class Meta:
|
||||||
model = Account
|
model = Account
|
||||||
exclude = ('id',)
|
exclude = ('id',)
|
||||||
|
@ -220,27 +216,28 @@ The `nested` option may be set to either `True`, `False`, or an integer value.
|
||||||
|
|
||||||
When serializing objects using a nested representation any occurances of recursion will be recognised, and will fall back to using a flat representation.
|
When serializing objects using a nested representation any occurances of recursion will be recognised, and will fall back to using a flat representation.
|
||||||
|
|
||||||
The `nested` option may also be set by passing it to the `serialize()` method.
|
|
||||||
|
|
||||||
**[TODO: Possibly only allow .serialize(nested=…) in FixtureSerializer]**
|
|
||||||
|
|
||||||
## Customising the default fields used by a ModelSerializer
|
## Customising the default fields used by a ModelSerializer
|
||||||
|
|
||||||
class AccountSerializer(ModelSerializer):
|
|
||||||
|
|
||||||
|
class AccountSerializer(serializers.ModelSerializer):
|
||||||
class Meta:
|
class Meta:
|
||||||
model = Account
|
model = Account
|
||||||
|
|
||||||
def get_pk_field(self, model_field):
|
def get_pk_field(self, model_field):
|
||||||
return Field(readonly=True)
|
return serializers.Field(readonly=True)
|
||||||
|
|
||||||
def get_nested_field(self, model_field):
|
def get_nested_field(self, model_field):
|
||||||
return ModelSerializer()
|
return serializers.ModelSerializer()
|
||||||
|
|
||||||
def get_related_field(self, model_field):
|
def get_related_field(self, model_field, to_many=False):
|
||||||
return NaturalKeyField()
|
queryset = model_field.rel.to._default_manager
|
||||||
|
if to_many:
|
||||||
|
return return serializers.ManyRelatedField(queryset=queryset)
|
||||||
|
return serializers.RelatedField(queryset=queryset)
|
||||||
|
|
||||||
def get_field(self, model_field):
|
def get_field(self, model_field):
|
||||||
return Field()
|
return serializers.ModelField(model_field=model_field)
|
||||||
|
|
||||||
|
|
||||||
[cite]: https://groups.google.com/d/topic/django-users/sVFaOfQi4wY/discussion
|
[cite]: https://groups.google.com/d/topic/django-users/sVFaOfQi4wY/discussion
|
||||||
|
|
|
@ -323,7 +323,9 @@ class ModelSerializer(Serializer):
|
||||||
elif model_field.rel and nested:
|
elif model_field.rel and nested:
|
||||||
field = self.get_nested_field(model_field)
|
field = self.get_nested_field(model_field)
|
||||||
elif model_field.rel:
|
elif model_field.rel:
|
||||||
field = self.get_related_field(model_field)
|
to_many = isinstance(model_field,
|
||||||
|
models.fields.related.ManyToManyField)
|
||||||
|
field = self.get_related_field(model_field, to_many=to_many)
|
||||||
else:
|
else:
|
||||||
field = self.get_field(model_field)
|
field = self.get_field(model_field)
|
||||||
|
|
||||||
|
@ -345,14 +347,14 @@ class ModelSerializer(Serializer):
|
||||||
"""
|
"""
|
||||||
return ModelSerializer()
|
return ModelSerializer()
|
||||||
|
|
||||||
def get_related_field(self, model_field):
|
def get_related_field(self, model_field, to_many=False):
|
||||||
"""
|
"""
|
||||||
Creates a default instance of a flat relational field.
|
Creates a default instance of a flat relational field.
|
||||||
"""
|
"""
|
||||||
# TODO: filter queryset using:
|
# TODO: filter queryset using:
|
||||||
# .using(db).complex_filter(self.rel.limit_choices_to)
|
# .using(db).complex_filter(self.rel.limit_choices_to)
|
||||||
queryset = model_field.rel.to._default_manager
|
queryset = model_field.rel.to._default_manager
|
||||||
if isinstance(model_field, models.fields.related.ManyToManyField):
|
if to_many:
|
||||||
return ManyPrimaryKeyRelatedField(queryset=queryset)
|
return ManyPrimaryKeyRelatedField(queryset=queryset)
|
||||||
return PrimaryKeyRelatedField(queryset=queryset)
|
return PrimaryKeyRelatedField(queryset=queryset)
|
||||||
|
|
||||||
|
@ -446,7 +448,7 @@ class HyperlinkedModelSerializer(ModelSerializer):
|
||||||
def get_pk_field(self, model_field):
|
def get_pk_field(self, model_field):
|
||||||
return None
|
return None
|
||||||
|
|
||||||
def get_related_field(self, model_field):
|
def get_related_field(self, model_field, to_many):
|
||||||
"""
|
"""
|
||||||
Creates a default instance of a flat relational field.
|
Creates a default instance of a flat relational field.
|
||||||
"""
|
"""
|
||||||
|
@ -458,6 +460,6 @@ class HyperlinkedModelSerializer(ModelSerializer):
|
||||||
'queryset': queryset,
|
'queryset': queryset,
|
||||||
'view_name': self._get_default_view_name(rel)
|
'view_name': self._get_default_view_name(rel)
|
||||||
}
|
}
|
||||||
if isinstance(model_field, models.fields.related.ManyToManyField):
|
if to_many:
|
||||||
return ManyHyperlinkedRelatedField(**kwargs)
|
return ManyHyperlinkedRelatedField(**kwargs)
|
||||||
return HyperlinkedRelatedField(**kwargs)
|
return HyperlinkedRelatedField(**kwargs)
|
||||||
|
|
Loading…
Reference in New Issue
Block a user