[1.10.x] Fixed #24931 -- Made namespaced RegexURLResolvers populate callback strings.

Fixed a regression in 2f16ff5a6c.

Thanks Tim Graham for the review.

Backport of 625b8e9295 from master
This commit is contained in:
Markus Holtermann 2016-06-18 16:39:32 +02:00 committed by Tim Graham
parent 1213ef2b18
commit 31a789f646
5 changed files with 60 additions and 1 deletions

View File

@ -164,6 +164,7 @@ class RegexURLResolver(LocaleRegexProvider):
# urlpatterns # urlpatterns
self._callback_strs = set() self._callback_strs = set()
self._populated = False self._populated = False
self._populating = False
def __repr__(self): def __repr__(self):
if isinstance(self.urlconf_name, list) and len(self.urlconf_name): if isinstance(self.urlconf_name, list) and len(self.urlconf_name):
@ -177,6 +178,9 @@ class RegexURLResolver(LocaleRegexProvider):
) )
def _populate(self): def _populate(self):
if self._populating:
return
self._populating = True
lookups = MultiValueDict() lookups = MultiValueDict()
namespaces = {} namespaces = {}
apps = {} apps = {}
@ -209,6 +213,8 @@ class RegexURLResolver(LocaleRegexProvider):
namespaces[namespace] = (p_pattern + prefix, sub_pattern) namespaces[namespace] = (p_pattern + prefix, sub_pattern)
for app_name, namespace_list in pattern.app_dict.items(): for app_name, namespace_list in pattern.app_dict.items():
apps.setdefault(app_name, []).extend(namespace_list) apps.setdefault(app_name, []).extend(namespace_list)
if not pattern._populating:
pattern._populate()
self._callback_strs.update(pattern._callback_strs) self._callback_strs.update(pattern._callback_strs)
else: else:
bits = normalize(p_pattern) bits = normalize(p_pattern)
@ -219,6 +225,7 @@ class RegexURLResolver(LocaleRegexProvider):
self._namespace_dict[language_code] = namespaces self._namespace_dict[language_code] = namespaces
self._app_dict[language_code] = apps self._app_dict[language_code] = apps
self._populated = True self._populated = True
self._populating = False
@property @property
def reverse_dict(self): def reverse_dict(self):

View File

@ -0,0 +1,14 @@
from django.conf.urls import include, url
from django.contrib import admin
from . import views
backend_urls = ([
url(r'^something/$', views.XViewClass.as_view(), name='something'),
], 'backend')
urlpatterns = [
url(r'^admin/doc/', include('django.contrib.admindocs.urls')),
url(r'^admin/', admin.site.urls),
url(r'^api/backend/', include(backend_urls, namespace='backend')),
]

View File

@ -88,6 +88,12 @@ class AdminDocViewTests(TestDataMixin, AdminDocsTestCase):
# View docstring # View docstring
self.assertContains(response, 'Base view for admindocs views.') self.assertContains(response, 'Base view for admindocs views.')
@override_settings(ROOT_URLCONF='admin_docs.namespace_urls')
def test_namespaced_view_detail(self):
url = reverse('django-admindocs-views-detail', args=['admin_docs.views.XViewClass'])
response = self.client.get(url)
self.assertContains(response, '<h1>admin_docs.views.XViewClass</h1>')
def test_view_detail_illegal_import(self): def test_view_detail_illegal_import(self):
""" """
#23601 - Ensure the view exists in the URLconf. #23601 - Ensure the view exists in the URLconf.

View File

@ -0,0 +1,25 @@
from django.conf.urls import include, url
from django.views import View
def view1(request):
pass
def view2(request):
pass
class View3(View):
pass
nested = ([
url(r'^view1/$', view1, name='view1'),
url(r'^view3/$', View3.as_view(), name='view3'),
], 'backend')
urlpatterns = [
url(r'^some/path/', include(nested, namespace='nested')),
url(r'^view2/$', view2, name='view2'),
]

View File

@ -428,6 +428,13 @@ class ResolverTests(unittest.TestCase):
'Wrong URL name. Expected "%s", got "%s".' % (e['name'], t.name) 'Wrong URL name. Expected "%s", got "%s".' % (e['name'], t.name)
) )
def test_namespaced_view_detail(self):
resolver = get_resolver('urlpatterns_reverse.nested_urls')
self.assertTrue(resolver._is_callback('urlpatterns_reverse.nested_urls.view1'))
self.assertTrue(resolver._is_callback('urlpatterns_reverse.nested_urls.view2'))
self.assertTrue(resolver._is_callback('urlpatterns_reverse.nested_urls.View3'))
self.assertFalse(resolver._is_callback('urlpatterns_reverse.nested_urls.blub'))
@override_settings(ROOT_URLCONF='urlpatterns_reverse.reverse_lazy_urls') @override_settings(ROOT_URLCONF='urlpatterns_reverse.reverse_lazy_urls')
class ReverseLazyTest(TestCase): class ReverseLazyTest(TestCase):