2012-11-22 03:20:49 +04:00
|
|
|
from __future__ import unicode_literals
|
2014-09-11 16:20:44 +04:00
|
|
|
import django
|
2012-12-18 01:59:51 +04:00
|
|
|
from django.db import models
|
2013-04-17 12:03:24 +04:00
|
|
|
from django.shortcuts import get_object_or_404
|
2012-09-28 17:28:50 +04:00
|
|
|
from django.test import TestCase
|
2014-08-19 20:06:55 +04:00
|
|
|
from django.utils import six
|
2013-05-18 12:36:09 +04:00
|
|
|
from rest_framework import generics, renderers, serializers, status
|
2013-06-28 20:17:39 +04:00
|
|
|
from rest_framework.test import APIRequestFactory
|
2014-03-02 15:40:30 +04:00
|
|
|
from tests.models import BasicModel, Comment, SlugBasedModel
|
2014-06-23 16:02:45 +04:00
|
|
|
from tests.models import ForeignKeySource, ForeignKeyTarget
|
2012-09-28 17:28:50 +04:00
|
|
|
|
2013-06-28 20:17:39 +04:00
|
|
|
factory = APIRequestFactory()
|
2012-09-28 17:28:50 +04:00
|
|
|
|
|
|
|
|
2014-08-29 15:35:53 +04:00
|
|
|
class BasicSerializer(serializers.ModelSerializer):
|
|
|
|
class Meta:
|
|
|
|
model = BasicModel
|
|
|
|
|
|
|
|
|
|
|
|
class ForeignKeySerializer(serializers.ModelSerializer):
|
|
|
|
class Meta:
|
|
|
|
model = ForeignKeySource
|
|
|
|
|
|
|
|
|
2012-10-03 12:26:15 +04:00
|
|
|
class RootView(generics.ListCreateAPIView):
|
2012-09-28 19:41:35 +04:00
|
|
|
"""
|
|
|
|
Example description for OPTIONS.
|
|
|
|
"""
|
2014-08-29 15:35:53 +04:00
|
|
|
queryset = BasicModel.objects.all()
|
|
|
|
serializer_class = BasicSerializer
|
2012-09-28 17:28:50 +04:00
|
|
|
|
|
|
|
|
2012-12-30 21:57:43 +04:00
|
|
|
class InstanceView(generics.RetrieveUpdateDestroyAPIView):
|
2012-09-28 19:41:35 +04:00
|
|
|
"""
|
|
|
|
Example description for OPTIONS.
|
|
|
|
"""
|
2014-09-02 20:41:23 +04:00
|
|
|
queryset = BasicModel.objects.exclude(text='filtered out')
|
2014-08-29 15:35:53 +04:00
|
|
|
serializer_class = BasicSerializer
|
2012-09-28 18:54:00 +04:00
|
|
|
|
|
|
|
|
2014-03-30 13:48:17 +04:00
|
|
|
class FKInstanceView(generics.RetrieveUpdateDestroyAPIView):
|
|
|
|
"""
|
|
|
|
FK: example description for OPTIONS.
|
|
|
|
"""
|
2014-08-29 15:35:53 +04:00
|
|
|
queryset = ForeignKeySource.objects.all()
|
|
|
|
serializer_class = ForeignKeySerializer
|
2014-03-30 13:48:17 +04:00
|
|
|
|
|
|
|
|
2012-10-29 21:20:06 +04:00
|
|
|
class SlugSerializer(serializers.ModelSerializer):
|
2014-09-02 20:41:23 +04:00
|
|
|
slug = serializers.Field(read_only=True)
|
2012-10-29 21:20:06 +04:00
|
|
|
|
|
|
|
class Meta:
|
|
|
|
model = SlugBasedModel
|
2014-09-02 20:41:23 +04:00
|
|
|
fields = ('text', 'slug')
|
2012-10-29 21:20:06 +04:00
|
|
|
|
|
|
|
|
2012-10-28 22:35:50 +04:00
|
|
|
class SlugBasedInstanceView(InstanceView):
|
|
|
|
"""
|
|
|
|
A model with a slug-field.
|
|
|
|
"""
|
2014-08-29 15:35:53 +04:00
|
|
|
queryset = SlugBasedModel.objects.all()
|
2012-10-29 21:20:06 +04:00
|
|
|
serializer_class = SlugSerializer
|
2013-05-18 00:57:11 +04:00
|
|
|
lookup_field = 'slug'
|
2012-10-28 22:35:50 +04:00
|
|
|
|
|
|
|
|
2012-09-28 18:54:00 +04:00
|
|
|
class TestRootView(TestCase):
|
2012-09-28 17:28:50 +04:00
|
|
|
def setUp(self):
|
2012-09-28 18:54:00 +04:00
|
|
|
"""
|
2013-02-25 17:44:19 +04:00
|
|
|
Create 3 BasicModel instances.
|
2012-09-28 18:54:00 +04:00
|
|
|
"""
|
2012-09-28 17:28:50 +04:00
|
|
|
items = ['foo', 'bar', 'baz']
|
|
|
|
for item in items:
|
|
|
|
BasicModel(text=item).save()
|
|
|
|
self.objects = BasicModel.objects
|
|
|
|
self.data = [
|
|
|
|
{'id': obj.id, 'text': obj.text}
|
|
|
|
for obj in self.objects.all()
|
|
|
|
]
|
2012-09-28 19:23:46 +04:00
|
|
|
self.view = RootView.as_view()
|
2012-09-28 17:28:50 +04:00
|
|
|
|
|
|
|
def test_get_root_view(self):
|
2012-09-28 18:54:00 +04:00
|
|
|
"""
|
2012-10-03 12:26:15 +04:00
|
|
|
GET requests to ListCreateAPIView should return list of objects.
|
2012-09-28 18:54:00 +04:00
|
|
|
"""
|
2012-09-28 17:28:50 +04:00
|
|
|
request = factory.get('/')
|
2013-03-08 21:36:43 +04:00
|
|
|
with self.assertNumQueries(1):
|
|
|
|
response = self.view(request).render()
|
2013-02-28 01:15:00 +04:00
|
|
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
|
|
|
self.assertEqual(response.data, self.data)
|
2012-09-28 18:54:00 +04:00
|
|
|
|
|
|
|
def test_post_root_view(self):
|
|
|
|
"""
|
2012-10-03 12:26:15 +04:00
|
|
|
POST requests to ListCreateAPIView should create a new object.
|
2012-09-28 18:54:00 +04:00
|
|
|
"""
|
2013-06-28 20:17:39 +04:00
|
|
|
data = {'text': 'foobar'}
|
|
|
|
request = factory.post('/', data, format='json')
|
2013-03-08 21:36:43 +04:00
|
|
|
with self.assertNumQueries(1):
|
|
|
|
response = self.view(request).render()
|
2013-02-28 01:15:00 +04:00
|
|
|
self.assertEqual(response.status_code, status.HTTP_201_CREATED)
|
|
|
|
self.assertEqual(response.data, {'id': 4, 'text': 'foobar'})
|
2012-09-28 18:54:00 +04:00
|
|
|
created = self.objects.get(id=4)
|
2013-02-28 01:15:00 +04:00
|
|
|
self.assertEqual(created.text, 'foobar')
|
2012-09-28 18:54:00 +04:00
|
|
|
|
2012-09-28 19:23:46 +04:00
|
|
|
def test_put_root_view(self):
|
|
|
|
"""
|
2012-10-03 12:26:15 +04:00
|
|
|
PUT requests to ListCreateAPIView should not be allowed
|
2012-09-28 19:23:46 +04:00
|
|
|
"""
|
2013-06-28 20:17:39 +04:00
|
|
|
data = {'text': 'foobar'}
|
|
|
|
request = factory.put('/', data, format='json')
|
2013-03-08 21:36:43 +04:00
|
|
|
with self.assertNumQueries(0):
|
|
|
|
response = self.view(request).render()
|
2013-02-28 01:15:00 +04:00
|
|
|
self.assertEqual(response.status_code, status.HTTP_405_METHOD_NOT_ALLOWED)
|
|
|
|
self.assertEqual(response.data, {"detail": "Method 'PUT' not allowed."})
|
2012-09-28 19:23:46 +04:00
|
|
|
|
|
|
|
def test_delete_root_view(self):
|
|
|
|
"""
|
2012-10-03 12:26:15 +04:00
|
|
|
DELETE requests to ListCreateAPIView should not be allowed
|
2012-09-28 19:23:46 +04:00
|
|
|
"""
|
|
|
|
request = factory.delete('/')
|
2013-03-08 21:36:43 +04:00
|
|
|
with self.assertNumQueries(0):
|
|
|
|
response = self.view(request).render()
|
2013-02-28 01:15:00 +04:00
|
|
|
self.assertEqual(response.status_code, status.HTTP_405_METHOD_NOT_ALLOWED)
|
|
|
|
self.assertEqual(response.data, {"detail": "Method 'DELETE' not allowed."})
|
2012-09-28 19:41:35 +04:00
|
|
|
|
2014-09-02 20:41:23 +04:00
|
|
|
# def test_options_root_view(self):
|
|
|
|
# """
|
|
|
|
# OPTIONS requests to ListCreateAPIView should return metadata
|
|
|
|
# """
|
|
|
|
# request = factory.options('/')
|
|
|
|
# with self.assertNumQueries(0):
|
|
|
|
# response = self.view(request).render()
|
|
|
|
# expected = {
|
|
|
|
# 'parses': [
|
|
|
|
# 'application/json',
|
|
|
|
# 'application/x-www-form-urlencoded',
|
|
|
|
# 'multipart/form-data'
|
|
|
|
# ],
|
|
|
|
# 'renders': [
|
|
|
|
# 'application/json',
|
|
|
|
# 'text/html'
|
|
|
|
# ],
|
|
|
|
# 'name': 'Root',
|
|
|
|
# 'description': 'Example description for OPTIONS.',
|
|
|
|
# 'actions': {
|
|
|
|
# 'POST': {
|
|
|
|
# 'text': {
|
|
|
|
# 'max_length': 100,
|
|
|
|
# 'read_only': False,
|
|
|
|
# 'required': True,
|
|
|
|
# 'type': 'string',
|
|
|
|
# "label": "Text comes here",
|
|
|
|
# "help_text": "Text description."
|
|
|
|
# },
|
|
|
|
# 'id': {
|
|
|
|
# 'read_only': True,
|
|
|
|
# 'required': False,
|
|
|
|
# 'type': 'integer',
|
|
|
|
# 'label': 'ID',
|
|
|
|
# },
|
|
|
|
# }
|
|
|
|
# }
|
|
|
|
# }
|
|
|
|
# self.assertEqual(response.status_code, status.HTTP_200_OK)
|
|
|
|
# self.assertEqual(response.data, expected)
|
2012-09-28 19:23:46 +04:00
|
|
|
|
2012-10-02 18:37:13 +04:00
|
|
|
def test_post_cannot_set_id(self):
|
|
|
|
"""
|
|
|
|
POST requests to create a new object should not be able to set the id.
|
|
|
|
"""
|
2013-06-28 20:17:39 +04:00
|
|
|
data = {'id': 999, 'text': 'foobar'}
|
|
|
|
request = factory.post('/', data, format='json')
|
2013-03-08 21:36:43 +04:00
|
|
|
with self.assertNumQueries(1):
|
|
|
|
response = self.view(request).render()
|
2013-02-28 01:15:00 +04:00
|
|
|
self.assertEqual(response.status_code, status.HTTP_201_CREATED)
|
|
|
|
self.assertEqual(response.data, {'id': 4, 'text': 'foobar'})
|
2012-10-02 18:37:13 +04:00
|
|
|
created = self.objects.get(id=4)
|
2013-02-28 01:15:00 +04:00
|
|
|
self.assertEqual(created.text, 'foobar')
|
2012-10-02 18:37:13 +04:00
|
|
|
|
2012-09-28 18:54:00 +04:00
|
|
|
|
2014-09-11 16:20:44 +04:00
|
|
|
EXPECTED_QUERYS_FOR_PUT = 3 if django.VERSION < (1, 6) else 2
|
|
|
|
|
|
|
|
|
2012-09-28 18:54:00 +04:00
|
|
|
class TestInstanceView(TestCase):
|
|
|
|
def setUp(self):
|
|
|
|
"""
|
|
|
|
Create 3 BasicModel intances.
|
|
|
|
"""
|
2013-11-19 18:49:31 +04:00
|
|
|
items = ['foo', 'bar', 'baz', 'filtered out']
|
2012-09-28 18:54:00 +04:00
|
|
|
for item in items:
|
|
|
|
BasicModel(text=item).save()
|
2013-11-19 18:49:31 +04:00
|
|
|
self.objects = BasicModel.objects.exclude(text='filtered out')
|
2012-09-28 18:54:00 +04:00
|
|
|
self.data = [
|
|
|
|
{'id': obj.id, 'text': obj.text}
|
|
|
|
for obj in self.objects.all()
|
|
|
|
]
|
2012-09-28 19:23:46 +04:00
|
|
|
self.view = InstanceView.as_view()
|
2012-10-28 22:35:50 +04:00
|
|
|
self.slug_based_view = SlugBasedInstanceView.as_view()
|
2012-09-28 18:54:00 +04:00
|
|
|
|
|
|
|
def test_get_instance_view(self):
|
|
|
|
"""
|
2012-10-03 12:26:15 +04:00
|
|
|
GET requests to RetrieveUpdateDestroyAPIView should return a single object.
|
2012-09-28 18:54:00 +04:00
|
|
|
"""
|
|
|
|
request = factory.get('/1')
|
2013-03-08 21:36:43 +04:00
|
|
|
with self.assertNumQueries(1):
|
|
|
|
response = self.view(request, pk=1).render()
|
2013-02-28 01:15:00 +04:00
|
|
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
|
|
|
self.assertEqual(response.data, self.data[0])
|
2012-09-28 18:54:00 +04:00
|
|
|
|
2012-09-28 19:23:46 +04:00
|
|
|
def test_post_instance_view(self):
|
|
|
|
"""
|
2012-10-03 12:26:15 +04:00
|
|
|
POST requests to RetrieveUpdateDestroyAPIView should not be allowed
|
2012-09-28 19:23:46 +04:00
|
|
|
"""
|
2013-06-28 20:17:39 +04:00
|
|
|
data = {'text': 'foobar'}
|
|
|
|
request = factory.post('/', data, format='json')
|
2013-03-08 21:36:43 +04:00
|
|
|
with self.assertNumQueries(0):
|
|
|
|
response = self.view(request).render()
|
2013-02-28 01:15:00 +04:00
|
|
|
self.assertEqual(response.status_code, status.HTTP_405_METHOD_NOT_ALLOWED)
|
|
|
|
self.assertEqual(response.data, {"detail": "Method 'POST' not allowed."})
|
2012-09-28 19:23:46 +04:00
|
|
|
|
2012-09-28 18:54:00 +04:00
|
|
|
def test_put_instance_view(self):
|
|
|
|
"""
|
2012-10-03 12:26:15 +04:00
|
|
|
PUT requests to RetrieveUpdateDestroyAPIView should update an object.
|
2012-09-28 18:54:00 +04:00
|
|
|
"""
|
2013-06-28 20:17:39 +04:00
|
|
|
data = {'text': 'foobar'}
|
|
|
|
request = factory.put('/1', data, format='json')
|
2014-09-11 16:20:44 +04:00
|
|
|
with self.assertNumQueries(EXPECTED_QUERYS_FOR_PUT):
|
2013-03-08 21:36:43 +04:00
|
|
|
response = self.view(request, pk='1').render()
|
2013-02-28 01:15:00 +04:00
|
|
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
2014-09-02 20:41:23 +04:00
|
|
|
self.assertEqual(dict(response.data), {'id': 1, 'text': 'foobar'})
|
2012-09-28 18:54:00 +04:00
|
|
|
updated = self.objects.get(id=1)
|
2013-02-28 01:15:00 +04:00
|
|
|
self.assertEqual(updated.text, 'foobar')
|
2012-09-28 18:54:00 +04:00
|
|
|
|
2012-12-20 09:28:01 +04:00
|
|
|
def test_patch_instance_view(self):
|
|
|
|
"""
|
|
|
|
PATCH requests to RetrieveUpdateDestroyAPIView should update an object.
|
|
|
|
"""
|
2013-06-28 20:17:39 +04:00
|
|
|
data = {'text': 'foobar'}
|
|
|
|
request = factory.patch('/1', data, format='json')
|
2012-12-20 09:28:01 +04:00
|
|
|
|
2014-09-11 16:20:44 +04:00
|
|
|
with self.assertNumQueries(EXPECTED_QUERYS_FOR_PUT):
|
2013-03-08 21:36:43 +04:00
|
|
|
response = self.view(request, pk=1).render()
|
2013-02-28 01:15:00 +04:00
|
|
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
|
|
|
self.assertEqual(response.data, {'id': 1, 'text': 'foobar'})
|
2012-12-20 09:28:01 +04:00
|
|
|
updated = self.objects.get(id=1)
|
2013-02-28 01:15:00 +04:00
|
|
|
self.assertEqual(updated.text, 'foobar')
|
2012-12-16 23:49:18 +04:00
|
|
|
|
2012-09-28 18:54:00 +04:00
|
|
|
def test_delete_instance_view(self):
|
|
|
|
"""
|
2012-10-03 12:26:15 +04:00
|
|
|
DELETE requests to RetrieveUpdateDestroyAPIView should delete an object.
|
2012-09-28 18:54:00 +04:00
|
|
|
"""
|
|
|
|
request = factory.delete('/1')
|
2013-03-08 21:36:43 +04:00
|
|
|
with self.assertNumQueries(2):
|
|
|
|
response = self.view(request, pk=1).render()
|
2013-02-28 01:15:00 +04:00
|
|
|
self.assertEqual(response.status_code, status.HTTP_204_NO_CONTENT)
|
|
|
|
self.assertEqual(response.content, six.b(''))
|
2012-09-28 18:54:00 +04:00
|
|
|
ids = [obj.id for obj in self.objects.all()]
|
2013-02-28 01:15:00 +04:00
|
|
|
self.assertEqual(ids, [2, 3])
|
2012-09-28 19:41:35 +04:00
|
|
|
|
2014-09-02 20:41:23 +04:00
|
|
|
# def test_options_instance_view(self):
|
|
|
|
# """
|
|
|
|
# OPTIONS requests to RetrieveUpdateDestroyAPIView should return metadata
|
|
|
|
# """
|
|
|
|
# request = factory.options('/1')
|
|
|
|
# with self.assertNumQueries(1):
|
|
|
|
# response = self.view(request, pk=1).render()
|
|
|
|
# expected = {
|
|
|
|
# 'parses': [
|
|
|
|
# 'application/json',
|
|
|
|
# 'application/x-www-form-urlencoded',
|
|
|
|
# 'multipart/form-data'
|
|
|
|
# ],
|
|
|
|
# 'renders': [
|
|
|
|
# 'application/json',
|
|
|
|
# 'text/html'
|
|
|
|
# ],
|
|
|
|
# 'name': 'Instance',
|
|
|
|
# 'description': 'Example description for OPTIONS.',
|
|
|
|
# 'actions': {
|
|
|
|
# 'PUT': {
|
|
|
|
# 'text': {
|
|
|
|
# 'max_length': 100,
|
|
|
|
# 'read_only': False,
|
|
|
|
# 'required': True,
|
|
|
|
# 'type': 'string',
|
|
|
|
# 'label': 'Text comes here',
|
|
|
|
# 'help_text': 'Text description.'
|
|
|
|
# },
|
|
|
|
# 'id': {
|
|
|
|
# 'read_only': True,
|
|
|
|
# 'required': False,
|
|
|
|
# 'type': 'integer',
|
|
|
|
# 'label': 'ID',
|
|
|
|
# },
|
|
|
|
# }
|
|
|
|
# }
|
|
|
|
# }
|
|
|
|
# self.assertEqual(response.status_code, status.HTTP_200_OK)
|
|
|
|
# self.assertEqual(response.data, expected)
|
|
|
|
|
|
|
|
# def test_options_before_instance_create(self):
|
|
|
|
# """
|
|
|
|
# OPTIONS requests to RetrieveUpdateDestroyAPIView should return metadata
|
|
|
|
# before the instance has been created
|
|
|
|
# """
|
|
|
|
# request = factory.options('/999')
|
|
|
|
# with self.assertNumQueries(1):
|
|
|
|
# response = self.view(request, pk=999).render()
|
|
|
|
# expected = {
|
|
|
|
# 'parses': [
|
|
|
|
# 'application/json',
|
|
|
|
# 'application/x-www-form-urlencoded',
|
|
|
|
# 'multipart/form-data'
|
|
|
|
# ],
|
|
|
|
# 'renders': [
|
|
|
|
# 'application/json',
|
|
|
|
# 'text/html'
|
|
|
|
# ],
|
|
|
|
# 'name': 'Instance',
|
|
|
|
# 'description': 'Example description for OPTIONS.',
|
|
|
|
# 'actions': {
|
|
|
|
# 'PUT': {
|
|
|
|
# 'text': {
|
|
|
|
# 'max_length': 100,
|
|
|
|
# 'read_only': False,
|
|
|
|
# 'required': True,
|
|
|
|
# 'type': 'string',
|
|
|
|
# 'label': 'Text comes here',
|
|
|
|
# 'help_text': 'Text description.'
|
|
|
|
# },
|
|
|
|
# 'id': {
|
|
|
|
# 'read_only': True,
|
|
|
|
# 'required': False,
|
|
|
|
# 'type': 'integer',
|
|
|
|
# 'label': 'ID',
|
|
|
|
# },
|
|
|
|
# }
|
|
|
|
# }
|
|
|
|
# }
|
|
|
|
# self.assertEqual(response.status_code, status.HTTP_200_OK)
|
|
|
|
# self.assertEqual(response.data, expected)
|
2012-10-02 18:39:41 +04:00
|
|
|
|
2013-05-28 14:57:11 +04:00
|
|
|
def test_get_instance_view_incorrect_arg(self):
|
|
|
|
"""
|
|
|
|
GET requests with an incorrect pk type, should raise 404, not 500.
|
|
|
|
Regression test for #890.
|
|
|
|
"""
|
|
|
|
request = factory.get('/a')
|
|
|
|
with self.assertNumQueries(0):
|
|
|
|
response = self.view(request, pk='a').render()
|
|
|
|
self.assertEqual(response.status_code, status.HTTP_404_NOT_FOUND)
|
|
|
|
|
2012-10-02 18:39:41 +04:00
|
|
|
def test_put_cannot_set_id(self):
|
|
|
|
"""
|
2012-10-28 22:35:50 +04:00
|
|
|
PUT requests to create a new object should not be able to set the id.
|
2012-10-02 18:39:41 +04:00
|
|
|
"""
|
2013-06-28 20:17:39 +04:00
|
|
|
data = {'id': 999, 'text': 'foobar'}
|
|
|
|
request = factory.put('/1', data, format='json')
|
2014-09-11 16:20:44 +04:00
|
|
|
with self.assertNumQueries(EXPECTED_QUERYS_FOR_PUT):
|
2013-03-08 21:36:43 +04:00
|
|
|
response = self.view(request, pk=1).render()
|
2013-02-28 01:15:00 +04:00
|
|
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
|
|
|
self.assertEqual(response.data, {'id': 1, 'text': 'foobar'})
|
2012-10-02 18:39:41 +04:00
|
|
|
updated = self.objects.get(id=1)
|
2013-02-28 01:15:00 +04:00
|
|
|
self.assertEqual(updated.text, 'foobar')
|
2012-10-05 19:24:52 +04:00
|
|
|
|
|
|
|
def test_put_to_deleted_instance(self):
|
|
|
|
"""
|
2014-09-05 19:29:46 +04:00
|
|
|
PUT requests to RetrieveUpdateDestroyAPIView should return 404 if
|
|
|
|
an object does not currently exist.
|
2012-10-05 19:24:52 +04:00
|
|
|
"""
|
|
|
|
self.objects.get(id=1).delete()
|
2013-06-28 20:17:39 +04:00
|
|
|
data = {'text': 'foobar'}
|
|
|
|
request = factory.put('/1', data, format='json')
|
2014-09-05 19:29:46 +04:00
|
|
|
with self.assertNumQueries(1):
|
2013-03-08 21:36:43 +04:00
|
|
|
response = self.view(request, pk=1).render()
|
2014-09-05 19:29:46 +04:00
|
|
|
self.assertEqual(response.status_code, status.HTTP_404_NOT_FOUND)
|
2012-10-28 22:35:50 +04:00
|
|
|
|
2013-11-19 18:49:31 +04:00
|
|
|
def test_put_to_filtered_out_instance(self):
|
|
|
|
"""
|
|
|
|
PUT requests to an URL of instance which is filtered out should not be
|
|
|
|
able to create new objects.
|
|
|
|
"""
|
|
|
|
data = {'text': 'foo'}
|
|
|
|
filtered_out_pk = BasicModel.objects.filter(text='filtered out')[0].pk
|
|
|
|
request = factory.put('/{0}'.format(filtered_out_pk), data, format='json')
|
|
|
|
response = self.view(request, pk=filtered_out_pk).render()
|
2014-09-05 19:29:46 +04:00
|
|
|
self.assertEqual(response.status_code, status.HTTP_404_NOT_FOUND)
|
2012-10-08 15:52:56 +04:00
|
|
|
|
2013-08-13 15:26:30 +04:00
|
|
|
def test_patch_cannot_create_an_object(self):
|
|
|
|
"""
|
|
|
|
PATCH requests should not be able to create objects.
|
|
|
|
"""
|
|
|
|
data = {'text': 'foobar'}
|
|
|
|
request = factory.patch('/999', data, format='json')
|
|
|
|
with self.assertNumQueries(1):
|
|
|
|
response = self.view(request, pk=999).render()
|
|
|
|
self.assertEqual(response.status_code, status.HTTP_404_NOT_FOUND)
|
|
|
|
self.assertFalse(self.objects.filter(id=999).exists())
|
|
|
|
|
2013-04-17 12:03:24 +04:00
|
|
|
|
2014-03-30 13:48:17 +04:00
|
|
|
class TestFKInstanceView(TestCase):
|
|
|
|
def setUp(self):
|
|
|
|
"""
|
|
|
|
Create 3 BasicModel instances.
|
|
|
|
"""
|
|
|
|
items = ['foo', 'bar', 'baz']
|
|
|
|
for item in items:
|
|
|
|
t = ForeignKeyTarget(name=item)
|
|
|
|
t.save()
|
|
|
|
ForeignKeySource(name='source_' + item, target=t).save()
|
|
|
|
|
|
|
|
self.objects = ForeignKeySource.objects
|
|
|
|
self.data = [
|
|
|
|
{'id': obj.id, 'name': obj.name}
|
|
|
|
for obj in self.objects.all()
|
|
|
|
]
|
|
|
|
self.view = FKInstanceView.as_view()
|
|
|
|
|
2014-09-02 20:41:23 +04:00
|
|
|
# def test_options_root_view(self):
|
|
|
|
# """
|
|
|
|
# OPTIONS requests to ListCreateAPIView should return metadata
|
|
|
|
# """
|
|
|
|
# request = factory.options('/999')
|
|
|
|
# with self.assertNumQueries(1):
|
|
|
|
# response = self.view(request, pk=999).render()
|
|
|
|
# expected = {
|
|
|
|
# 'name': 'Fk Instance',
|
|
|
|
# 'description': 'FK: example description for OPTIONS.',
|
|
|
|
# 'renders': [
|
|
|
|
# 'application/json',
|
|
|
|
# 'text/html'
|
|
|
|
# ],
|
|
|
|
# 'parses': [
|
|
|
|
# 'application/json',
|
|
|
|
# 'application/x-www-form-urlencoded',
|
|
|
|
# 'multipart/form-data'
|
|
|
|
# ],
|
|
|
|
# 'actions': {
|
|
|
|
# 'PUT': {
|
|
|
|
# 'id': {
|
|
|
|
# 'type': 'integer',
|
|
|
|
# 'required': False,
|
|
|
|
# 'read_only': True,
|
|
|
|
# 'label': 'ID'
|
|
|
|
# },
|
|
|
|
# 'name': {
|
|
|
|
# 'type': 'string',
|
|
|
|
# 'required': True,
|
|
|
|
# 'read_only': False,
|
|
|
|
# 'label': 'name',
|
|
|
|
# 'max_length': 100
|
|
|
|
# },
|
|
|
|
# 'target': {
|
|
|
|
# 'type': 'field',
|
|
|
|
# 'required': True,
|
|
|
|
# 'read_only': False,
|
|
|
|
# 'label': 'Target',
|
|
|
|
# 'help_text': 'Target'
|
|
|
|
# }
|
|
|
|
# }
|
|
|
|
# }
|
|
|
|
# }
|
|
|
|
# self.assertEqual(response.status_code, status.HTTP_200_OK)
|
|
|
|
# self.assertEqual(response.data, expected)
|
2014-03-30 13:48:17 +04:00
|
|
|
|
|
|
|
|
2013-04-17 12:03:24 +04:00
|
|
|
class TestOverriddenGetObject(TestCase):
|
2013-04-14 20:30:44 +04:00
|
|
|
"""
|
|
|
|
Test cases for a RetrieveUpdateDestroyAPIView that does NOT use the
|
|
|
|
queryset/model mechanism but instead overrides get_object()
|
|
|
|
"""
|
|
|
|
def setUp(self):
|
|
|
|
"""
|
|
|
|
Create 3 BasicModel intances.
|
|
|
|
"""
|
|
|
|
items = ['foo', 'bar', 'baz']
|
|
|
|
for item in items:
|
|
|
|
BasicModel(text=item).save()
|
|
|
|
self.objects = BasicModel.objects
|
|
|
|
self.data = [
|
|
|
|
{'id': obj.id, 'text': obj.text}
|
|
|
|
for obj in self.objects.all()
|
|
|
|
]
|
|
|
|
|
2013-04-17 12:03:24 +04:00
|
|
|
class OverriddenGetObjectView(generics.RetrieveUpdateDestroyAPIView):
|
|
|
|
"""
|
|
|
|
Example detail view for override of get_object().
|
|
|
|
"""
|
2014-08-29 15:35:53 +04:00
|
|
|
serializer_class = BasicSerializer
|
2013-04-14 20:30:44 +04:00
|
|
|
|
2013-04-17 12:03:24 +04:00
|
|
|
def get_object(self):
|
|
|
|
pk = int(self.kwargs['pk'])
|
|
|
|
return get_object_or_404(BasicModel.objects.all(), id=pk)
|
2013-04-14 20:30:44 +04:00
|
|
|
|
2013-04-17 12:03:24 +04:00
|
|
|
self.view = OverriddenGetObjectView.as_view()
|
2013-04-14 20:30:44 +04:00
|
|
|
|
2013-04-17 12:03:24 +04:00
|
|
|
def test_overridden_get_object_view(self):
|
2013-04-14 20:30:44 +04:00
|
|
|
"""
|
2013-04-17 12:03:24 +04:00
|
|
|
GET requests to RetrieveUpdateDestroyAPIView should return a single object.
|
2013-04-14 20:30:44 +04:00
|
|
|
"""
|
2013-04-17 12:03:24 +04:00
|
|
|
request = factory.get('/1')
|
|
|
|
with self.assertNumQueries(1):
|
2013-04-14 20:30:44 +04:00
|
|
|
response = self.view(request, pk=1).render()
|
|
|
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
2013-04-17 12:03:24 +04:00
|
|
|
self.assertEqual(response.data, self.data[0])
|
2013-04-14 20:30:44 +04:00
|
|
|
|
2012-10-08 15:52:56 +04:00
|
|
|
|
|
|
|
# Regression test for #285
|
|
|
|
|
|
|
|
class CommentSerializer(serializers.ModelSerializer):
|
|
|
|
class Meta:
|
|
|
|
model = Comment
|
|
|
|
exclude = ('created',)
|
|
|
|
|
|
|
|
|
|
|
|
class CommentView(generics.ListCreateAPIView):
|
|
|
|
serializer_class = CommentSerializer
|
|
|
|
model = Comment
|
|
|
|
|
|
|
|
|
|
|
|
class TestCreateModelWithAutoNowAddField(TestCase):
|
|
|
|
def setUp(self):
|
|
|
|
self.objects = Comment.objects
|
|
|
|
self.view = CommentView.as_view()
|
|
|
|
|
|
|
|
def test_create_model_with_auto_now_add_field(self):
|
|
|
|
"""
|
|
|
|
Regression test for #285
|
|
|
|
|
|
|
|
https://github.com/tomchristie/django-rest-framework/issues/285
|
|
|
|
"""
|
2013-06-28 20:17:39 +04:00
|
|
|
data = {'email': 'foobar@example.com', 'content': 'foobar'}
|
|
|
|
request = factory.post('/', data, format='json')
|
2012-10-08 15:52:56 +04:00
|
|
|
response = self.view(request).render()
|
2013-02-28 01:15:00 +04:00
|
|
|
self.assertEqual(response.status_code, status.HTTP_201_CREATED)
|
2012-10-08 15:52:56 +04:00
|
|
|
created = self.objects.get(id=1)
|
2013-02-28 01:15:00 +04:00
|
|
|
self.assertEqual(created.content, 'foobar')
|
2012-12-18 01:59:51 +04:00
|
|
|
|
|
|
|
|
2013-05-02 08:26:40 +04:00
|
|
|
# Test for particularly ugly regression with m2m in browsable API
|
2012-12-18 01:59:51 +04:00
|
|
|
class ClassB(models.Model):
|
|
|
|
name = models.CharField(max_length=255)
|
|
|
|
|
|
|
|
|
|
|
|
class ClassA(models.Model):
|
|
|
|
name = models.CharField(max_length=255)
|
|
|
|
childs = models.ManyToManyField(ClassB, blank=True, null=True)
|
|
|
|
|
|
|
|
|
|
|
|
class ClassASerializer(serializers.ModelSerializer):
|
2014-09-12 13:59:51 +04:00
|
|
|
childs = serializers.PrimaryKeyRelatedField(
|
|
|
|
many=True, queryset=ClassB.objects.all()
|
|
|
|
)
|
2012-12-18 01:59:51 +04:00
|
|
|
|
|
|
|
class Meta:
|
|
|
|
model = ClassA
|
|
|
|
|
|
|
|
|
|
|
|
class ExampleView(generics.ListCreateAPIView):
|
|
|
|
serializer_class = ClassASerializer
|
2014-08-29 15:35:53 +04:00
|
|
|
queryset = ClassA.objects.all()
|
2012-12-18 01:59:51 +04:00
|
|
|
|
|
|
|
|
|
|
|
class TestM2MBrowseableAPI(TestCase):
|
|
|
|
def test_m2m_in_browseable_api(self):
|
|
|
|
"""
|
2013-05-02 08:26:40 +04:00
|
|
|
Test for particularly ugly regression with m2m in browsable API
|
2012-12-18 01:59:51 +04:00
|
|
|
"""
|
|
|
|
request = factory.get('/', HTTP_ACCEPT='text/html')
|
|
|
|
view = ExampleView().as_view()
|
|
|
|
response = view(request).render()
|
2013-02-28 01:15:00 +04:00
|
|
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
2013-03-08 01:09:59 +04:00
|
|
|
|
|
|
|
|
|
|
|
class InclusiveFilterBackend(object):
|
|
|
|
def filter_queryset(self, request, queryset, view):
|
|
|
|
return queryset.filter(text='foo')
|
|
|
|
|
|
|
|
|
|
|
|
class ExclusiveFilterBackend(object):
|
|
|
|
def filter_queryset(self, request, queryset, view):
|
|
|
|
return queryset.filter(text='other')
|
|
|
|
|
|
|
|
|
2013-11-18 23:15:35 +04:00
|
|
|
class TwoFieldModel(models.Model):
|
|
|
|
field_a = models.CharField(max_length=100)
|
|
|
|
field_b = models.CharField(max_length=100)
|
|
|
|
|
|
|
|
|
|
|
|
class DynamicSerializerView(generics.ListCreateAPIView):
|
2014-08-29 15:35:53 +04:00
|
|
|
queryset = TwoFieldModel.objects.all()
|
2013-11-18 23:15:35 +04:00
|
|
|
renderer_classes = (renderers.BrowsableAPIRenderer, renderers.JSONRenderer)
|
|
|
|
|
|
|
|
def get_serializer_class(self):
|
|
|
|
if self.request.method == 'POST':
|
|
|
|
class DynamicSerializer(serializers.ModelSerializer):
|
|
|
|
class Meta:
|
|
|
|
model = TwoFieldModel
|
|
|
|
fields = ('field_b',)
|
2014-08-29 15:35:53 +04:00
|
|
|
else:
|
|
|
|
class DynamicSerializer(serializers.ModelSerializer):
|
|
|
|
class Meta:
|
|
|
|
model = TwoFieldModel
|
|
|
|
return DynamicSerializer
|
2013-11-18 23:15:35 +04:00
|
|
|
|
|
|
|
|
2013-03-08 01:09:59 +04:00
|
|
|
class TestFilterBackendAppliedToViews(TestCase):
|
|
|
|
|
|
|
|
def setUp(self):
|
|
|
|
"""
|
|
|
|
Create 3 BasicModel instances to filter on.
|
|
|
|
"""
|
|
|
|
items = ['foo', 'bar', 'baz']
|
|
|
|
for item in items:
|
|
|
|
BasicModel(text=item).save()
|
|
|
|
self.objects = BasicModel.objects
|
|
|
|
self.data = [
|
|
|
|
{'id': obj.id, 'text': obj.text}
|
|
|
|
for obj in self.objects.all()
|
|
|
|
]
|
|
|
|
|
|
|
|
def test_get_root_view_filters_by_name_with_filter_backend(self):
|
|
|
|
"""
|
|
|
|
GET requests to ListCreateAPIView should return filtered list.
|
|
|
|
"""
|
2013-05-18 00:57:11 +04:00
|
|
|
root_view = RootView.as_view(filter_backends=(InclusiveFilterBackend,))
|
2013-03-08 01:09:59 +04:00
|
|
|
request = factory.get('/')
|
2013-05-18 00:57:11 +04:00
|
|
|
response = root_view(request).render()
|
2013-03-08 01:09:59 +04:00
|
|
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
|
|
|
self.assertEqual(len(response.data), 1)
|
|
|
|
self.assertEqual(response.data, [{'id': 1, 'text': 'foo'}])
|
|
|
|
|
|
|
|
def test_get_root_view_filters_out_all_models_with_exclusive_filter_backend(self):
|
|
|
|
"""
|
|
|
|
GET requests to ListCreateAPIView should return empty list when all models are filtered out.
|
|
|
|
"""
|
2013-05-18 00:57:11 +04:00
|
|
|
root_view = RootView.as_view(filter_backends=(ExclusiveFilterBackend,))
|
2013-03-08 01:09:59 +04:00
|
|
|
request = factory.get('/')
|
2013-05-18 00:57:11 +04:00
|
|
|
response = root_view(request).render()
|
2013-03-08 01:09:59 +04:00
|
|
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
|
|
|
self.assertEqual(response.data, [])
|
|
|
|
|
|
|
|
def test_get_instance_view_filters_out_name_with_filter_backend(self):
|
|
|
|
"""
|
|
|
|
GET requests to RetrieveUpdateDestroyAPIView should raise 404 when model filtered out.
|
|
|
|
"""
|
2013-05-18 00:57:11 +04:00
|
|
|
instance_view = InstanceView.as_view(filter_backends=(ExclusiveFilterBackend,))
|
2013-03-08 01:09:59 +04:00
|
|
|
request = factory.get('/1')
|
2013-05-18 00:57:11 +04:00
|
|
|
response = instance_view(request, pk=1).render()
|
2013-03-08 01:09:59 +04:00
|
|
|
self.assertEqual(response.status_code, status.HTTP_404_NOT_FOUND)
|
|
|
|
self.assertEqual(response.data, {'detail': 'Not found'})
|
|
|
|
|
|
|
|
def test_get_instance_view_will_return_single_object_when_filter_does_not_exclude_it(self):
|
|
|
|
"""
|
|
|
|
GET requests to RetrieveUpdateDestroyAPIView should return a single object when not excluded
|
|
|
|
"""
|
2013-05-18 00:57:11 +04:00
|
|
|
instance_view = InstanceView.as_view(filter_backends=(InclusiveFilterBackend,))
|
2013-03-08 01:09:59 +04:00
|
|
|
request = factory.get('/1')
|
2013-05-18 00:57:11 +04:00
|
|
|
response = instance_view(request, pk=1).render()
|
2013-03-08 01:09:59 +04:00
|
|
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
|
|
|
self.assertEqual(response.data, {'id': 1, 'text': 'foo'})
|
2013-05-18 12:36:09 +04:00
|
|
|
|
|
|
|
def test_dynamic_serializer_form_in_browsable_api(self):
|
|
|
|
"""
|
|
|
|
GET requests to ListCreateAPIView should return filtered list.
|
|
|
|
"""
|
|
|
|
view = DynamicSerializerView.as_view()
|
|
|
|
request = factory.get('/')
|
|
|
|
response = view(request).render()
|
|
|
|
self.assertContains(response, 'field_b')
|
|
|
|
self.assertNotContains(response, 'field_a')
|