mirror of
https://github.com/encode/django-rest-framework.git
synced 2024-11-25 19:14:01 +03:00
Merge pull request #5192 from matteius/DRF-5135-one-to-one-pk
Special case for when OneToOneField is also primary key.
This commit is contained in:
commit
598e5877cd
|
@ -1159,6 +1159,11 @@ class ModelSerializer(Serializer):
|
||||||
field_class = field_mapping[model_field]
|
field_class = field_mapping[model_field]
|
||||||
field_kwargs = get_field_kwargs(field_name, model_field)
|
field_kwargs = get_field_kwargs(field_name, model_field)
|
||||||
|
|
||||||
|
# Special case to handle when a OneToOneField is also the primary key
|
||||||
|
if model_field.one_to_one and model_field.primary_key:
|
||||||
|
field_class = self.serializer_related_field
|
||||||
|
field_kwargs['queryset'] = model_field.related_model.objects
|
||||||
|
|
||||||
if 'choices' in field_kwargs:
|
if 'choices' in field_kwargs:
|
||||||
# Fields with choices get coerced into `ChoiceField`
|
# Fields with choices get coerced into `ChoiceField`
|
||||||
# instead of using their regular typed field.
|
# instead of using their regular typed field.
|
||||||
|
|
|
@ -88,3 +88,11 @@ class NullableOneToOneSource(RESTFrameworkModel):
|
||||||
target = models.OneToOneField(
|
target = models.OneToOneField(
|
||||||
OneToOneTarget, null=True, blank=True,
|
OneToOneTarget, null=True, blank=True,
|
||||||
related_name='nullable_source', on_delete=models.CASCADE)
|
related_name='nullable_source', on_delete=models.CASCADE)
|
||||||
|
|
||||||
|
|
||||||
|
class OneToOnePKSource(RESTFrameworkModel):
|
||||||
|
""" Test model where the primary key is a OneToOneField with another model. """
|
||||||
|
name = models.CharField(max_length=100)
|
||||||
|
target = models.OneToOneField(
|
||||||
|
OneToOneTarget, primary_key=True,
|
||||||
|
related_name='required_source', on_delete=models.CASCADE)
|
||||||
|
|
|
@ -6,8 +6,8 @@ from django.utils import six
|
||||||
from rest_framework import serializers
|
from rest_framework import serializers
|
||||||
from tests.models import (
|
from tests.models import (
|
||||||
ForeignKeySource, ForeignKeyTarget, ManyToManySource, ManyToManyTarget,
|
ForeignKeySource, ForeignKeyTarget, ManyToManySource, ManyToManyTarget,
|
||||||
NullableForeignKeySource, NullableOneToOneSource,
|
NullableForeignKeySource, NullableOneToOneSource, NullableUUIDForeignKeySource,
|
||||||
NullableUUIDForeignKeySource, OneToOneTarget, UUIDForeignKeyTarget
|
OneToOnePKSource, OneToOneTarget, UUIDForeignKeyTarget
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@ -63,6 +63,13 @@ class NullableOneToOneTargetSerializer(serializers.ModelSerializer):
|
||||||
fields = ('id', 'name', 'nullable_source')
|
fields = ('id', 'name', 'nullable_source')
|
||||||
|
|
||||||
|
|
||||||
|
class OneToOnePKSourceSerializer(serializers.ModelSerializer):
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = OneToOnePKSource
|
||||||
|
fields = '__all__'
|
||||||
|
|
||||||
|
|
||||||
# TODO: Add test that .data cannot be accessed prior to .is_valid
|
# TODO: Add test that .data cannot be accessed prior to .is_valid
|
||||||
|
|
||||||
class PKManyToManyTests(TestCase):
|
class PKManyToManyTests(TestCase):
|
||||||
|
@ -486,3 +493,51 @@ class PKNullableOneToOneTests(TestCase):
|
||||||
{'id': 2, 'name': 'target-2', 'nullable_source': 1},
|
{'id': 2, 'name': 'target-2', 'nullable_source': 1},
|
||||||
]
|
]
|
||||||
assert serializer.data == expected
|
assert serializer.data == expected
|
||||||
|
|
||||||
|
|
||||||
|
class OneToOnePrimaryKeyTests(TestCase):
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
# Given: Some target models already exist
|
||||||
|
self.target = target = OneToOneTarget(name='target-1')
|
||||||
|
target.save()
|
||||||
|
self.alt_target = alt_target = OneToOneTarget(name='target-2')
|
||||||
|
alt_target.save()
|
||||||
|
|
||||||
|
def test_one_to_one_when_primary_key(self):
|
||||||
|
# When: Creating a Source pointing at the id of the second Target
|
||||||
|
target_pk = self.alt_target.id
|
||||||
|
source = OneToOnePKSourceSerializer(data={'name': 'source-2', 'target': target_pk})
|
||||||
|
# Then: The source is valid with the serializer
|
||||||
|
if not source.is_valid():
|
||||||
|
self.fail("Expected OneToOnePKTargetSerializer to be valid but had errors: {}".format(source.errors))
|
||||||
|
# Then: Saving the serializer creates a new object
|
||||||
|
new_source = source.save()
|
||||||
|
# Then: The new object has the same pk as the target object
|
||||||
|
self.assertEqual(new_source.pk, target_pk)
|
||||||
|
|
||||||
|
def test_one_to_one_when_primary_key_no_duplicates(self):
|
||||||
|
# When: Creating a Source pointing at the id of the second Target
|
||||||
|
target_pk = self.target.id
|
||||||
|
data = {'name': 'source-1', 'target': target_pk}
|
||||||
|
source = OneToOnePKSourceSerializer(data=data)
|
||||||
|
# Then: The source is valid with the serializer
|
||||||
|
self.assertTrue(source.is_valid())
|
||||||
|
# Then: Saving the serializer creates a new object
|
||||||
|
new_source = source.save()
|
||||||
|
# Then: The new object has the same pk as the target object
|
||||||
|
self.assertEqual(new_source.pk, target_pk)
|
||||||
|
# When: Trying to create a second object
|
||||||
|
second_source = OneToOnePKSourceSerializer(data=data)
|
||||||
|
self.assertFalse(second_source.is_valid())
|
||||||
|
expected = {'target': [u'one to one pk source with this target already exists.']}
|
||||||
|
self.assertDictEqual(second_source.errors, expected)
|
||||||
|
|
||||||
|
def test_one_to_one_when_primary_key_does_not_exist(self):
|
||||||
|
# Given: a target PK that does not exist
|
||||||
|
target_pk = self.target.pk + self.alt_target.pk
|
||||||
|
source = OneToOnePKSourceSerializer(data={'name': 'source-2', 'target': target_pk})
|
||||||
|
# Then: The source is not valid with the serializer
|
||||||
|
self.assertFalse(source.is_valid())
|
||||||
|
self.assertIn("Invalid pk", source.errors['target'][0])
|
||||||
|
self.assertIn("object does not exist", source.errors['target'][0])
|
||||||
|
|
Loading…
Reference in New Issue
Block a user