aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorTom Christie2015-03-23 11:35:21 +0000
committerTom Christie2015-03-23 11:35:21 +0000
commita90dbe3bd9ce88a6837d7d20f3e904f43d68c996 (patch)
tree4c4b2ef3b22e24645d3ce9314855368c239d64c1
parentf7cd7a1c1fca4c7f0e38c5ed1762f5f498bf37a3 (diff)
parentfac27853418699116304ad8d77270fe9a20873dc (diff)
downloaddjango-rest-framework-a90dbe3bd9ce88a6837d7d20f3e904f43d68c996.tar.bz2
Merge pull request #2724 from delinhabit/namespaced-versioning-non-api-reversal
Handle reversal of non-API view_name in HyperLinkedRelatedField
-rw-r--r--rest_framework/reverse.py9
-rw-r--r--tests/test_reverse.py27
-rw-r--r--tests/test_versioning.py44
3 files changed, 78 insertions, 2 deletions
diff --git a/rest_framework/reverse.py b/rest_framework/reverse.py
index a251d99d..e6d3f563 100644
--- a/rest_framework/reverse.py
+++ b/rest_framework/reverse.py
@@ -3,6 +3,7 @@ Provide urlresolver functions that return fully qualified URLs or view names
"""
from __future__ import unicode_literals
from django.core.urlresolvers import reverse as django_reverse
+from django.core.urlresolvers import NoReverseMatch
from django.utils import six
from django.utils.functional import lazy
@@ -15,7 +16,13 @@ def reverse(viewname, args=None, kwargs=None, request=None, format=None, **extra
"""
scheme = getattr(request, 'versioning_scheme', None)
if scheme is not None:
- return scheme.reverse(viewname, args, kwargs, request, format, **extra)
+ try:
+ return scheme.reverse(viewname, args, kwargs, request, format, **extra)
+ except NoReverseMatch:
+ # In case the versioning scheme reversal fails, fallback to the
+ # default implementation
+ pass
+
return _reverse(viewname, args, kwargs, request, format, **extra)
diff --git a/tests/test_reverse.py b/tests/test_reverse.py
index 675a9d5a..08c27023 100644
--- a/tests/test_reverse.py
+++ b/tests/test_reverse.py
@@ -1,5 +1,6 @@
from __future__ import unicode_literals
from django.conf.urls import patterns, url
+from django.core.urlresolvers import NoReverseMatch
from django.test import TestCase
from rest_framework.reverse import reverse
from rest_framework.test import APIRequestFactory
@@ -16,6 +17,18 @@ urlpatterns = patterns(
)
+class MockVersioningScheme(object):
+
+ def __init__(self, raise_error=False):
+ self.raise_error = raise_error
+
+ def reverse(self, *args, **kwargs):
+ if self.raise_error:
+ raise NoReverseMatch()
+
+ return 'http://scheme-reversed/view'
+
+
class ReverseTests(TestCase):
"""
Tests for fully qualified URLs when using `reverse`.
@@ -26,3 +39,17 @@ class ReverseTests(TestCase):
request = factory.get('/view')
url = reverse('view', request=request)
self.assertEqual(url, 'http://testserver/view')
+
+ def test_reverse_with_versioning_scheme(self):
+ request = factory.get('/view')
+ request.versioning_scheme = MockVersioningScheme()
+
+ url = reverse('view', request=request)
+ self.assertEqual(url, 'http://scheme-reversed/view')
+
+ def test_reverse_with_versioning_scheme_fallback_to_default_on_error(self):
+ request = factory.get('/view')
+ request.versioning_scheme = MockVersioningScheme(raise_error=True)
+
+ url = reverse('view', request=request)
+ self.assertEqual(url, 'http://testserver/view')
diff --git a/tests/test_versioning.py b/tests/test_versioning.py
index 90ad8afd..88ae56dd 100644
--- a/tests/test_versioning.py
+++ b/tests/test_versioning.py
@@ -7,6 +7,7 @@ from rest_framework.response import Response
from rest_framework.reverse import reverse
from rest_framework.test import APIRequestFactory, APITestCase
from rest_framework.versioning import NamespaceVersioning
+from rest_framework.relations import PKOnlyObject
import pytest
@@ -234,7 +235,7 @@ class TestInvalidVersion:
class TestHyperlinkedRelatedField(UsingURLPatterns, APITestCase):
included = [
- url(r'^namespaced/(?P<pk>\d+)/$', dummy_view, name='namespaced'),
+ url(r'^namespaced/(?P<pk>\d+)/$', dummy_pk_view, name='namespaced'),
]
urlpatterns = [
@@ -262,3 +263,44 @@ class TestHyperlinkedRelatedField(UsingURLPatterns, APITestCase):
assert self.field.to_internal_value('/v1/namespaced/3/') == 'object 3'
with pytest.raises(serializers.ValidationError):
self.field.to_internal_value('/v2/namespaced/3/')
+
+
+class TestNamespaceVersioningHyperlinkedRelatedFieldScheme(UsingURLPatterns, APITestCase):
+ included = [
+ url(r'^namespaced/(?P<pk>\d+)/$', dummy_pk_view, name='namespaced'),
+ ]
+
+ urlpatterns = [
+ url(r'^v1/', include(included, namespace='v1')),
+ url(r'^v2/', include(included, namespace='v2')),
+ url(r'^non-api/(?P<pk>\d+)/$', dummy_pk_view, name='non-api-view')
+ ]
+
+ def _create_field(self, view_name, version):
+ request = factory.get("/")
+ request.versioning_scheme = NamespaceVersioning()
+ request.version = version
+
+ field = serializers.HyperlinkedRelatedField(
+ view_name=view_name,
+ read_only=True)
+ field._context = {'request': request}
+ return field
+
+ def test_api_url_is_properly_reversed_with_v1(self):
+ field = self._create_field('namespaced', 'v1')
+ assert field.to_representation(PKOnlyObject(3)) == 'http://testserver/v1/namespaced/3/'
+
+ def test_api_url_is_properly_reversed_with_v2(self):
+ field = self._create_field('namespaced', 'v2')
+ assert field.to_representation(PKOnlyObject(5)) == 'http://testserver/v2/namespaced/5/'
+
+ def test_non_api_url_is_properly_reversed_regardless_of_the_version(self):
+ """
+ Regression test for #2711
+ """
+ field = self._create_field('non-api-view', 'v1')
+ assert field.to_representation(PKOnlyObject(10)) == 'http://testserver/non-api/10/'
+
+ field = self._create_field('non-api-view', 'v2')
+ assert field.to_representation(PKOnlyObject(10)) == 'http://testserver/non-api/10/'