mirror of
				https://github.com/django/django.git
				synced 2025-10-25 22:56:12 +00:00 
			
		
		
		
	[1.11.x] Fixed #27367 -- Doc'd and tested reversing of URLs with the same name.
Thanks Reinout van Rees for contributing to the patch.
Backport of 98bcc5d81b from master
			
			
This commit is contained in:
		
				
					committed by
					
						 Tim Graham
						Tim Graham
					
				
			
			
				
	
			
			
			
						parent
						
							ded0632d94
						
					
				
				
					commit
					c091614450
				
			| @@ -597,15 +597,28 @@ In order to perform URL reversing, you'll need to use **named URL patterns** | |||||||
| as done in the examples above. The string used for the URL name can contain any | as done in the examples above. The string used for the URL name can contain any | ||||||
| characters you like. You are not restricted to valid Python names. | characters you like. You are not restricted to valid Python names. | ||||||
|  |  | ||||||
| When you name your URL patterns, make sure you use names that are unlikely | When naming URL patterns, choose names that are unlikely to clash with other | ||||||
| to clash with any other application's choice of names. If you call your URL | applications' choice of names. If you call your URL pattern ``comment`` | ||||||
| pattern ``comment``, and another application does the same thing, there's | and another application does the same thing, the URL that | ||||||
| no guarantee which URL will be inserted into your template when you use | :func:`~django.urls.reverse()` finds depends on whichever pattern is last in | ||||||
| this name. | your project's ``urlpatterns`` list. | ||||||
|  |  | ||||||
| Putting a prefix on your URL names, perhaps derived from the application | Putting a prefix on your URL names, perhaps derived from the application | ||||||
| name, will decrease the chances of collision. We recommend something like | name (such as ``myapp-comment`` instead of ``comment``), decreases the chance | ||||||
| ``myapp-comment`` instead of ``comment``. | of collision. | ||||||
|  |  | ||||||
|  | You can deliberately choose the *same URL name* as another application if you | ||||||
|  | want to override a view. For example, a common use case is to override the | ||||||
|  | :class:`~django.contrib.auth.views.LoginView`. Parts of Django and most | ||||||
|  | third-party apps assume that this view has a URL pattern with the name | ||||||
|  | ``login``. If you have a custom login view and give its URL the name ``login``, | ||||||
|  | :func:`~django.urls.reverse()` will find your custom view as long as it's in | ||||||
|  | ``urlpatterns`` after ``django.contrib.auth.urls`` is included (if that's | ||||||
|  | included at all). | ||||||
|  |  | ||||||
|  | You may also use the same name for multiple URL patterns if they differ in | ||||||
|  | their arguments. In addition to the URL name, :func:`~django.urls.reverse()` | ||||||
|  | matches the number of arguments and the names of the keyword arguments. | ||||||
|  |  | ||||||
| .. _topics-http-defining-url-namespaces: | .. _topics-http-defining-url-namespaces: | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										17
									
								
								tests/urlpatterns_reverse/named_urls_conflict.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								tests/urlpatterns_reverse/named_urls_conflict.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | |||||||
|  | from django.conf.urls import url | ||||||
|  |  | ||||||
|  | from .views import empty_view | ||||||
|  |  | ||||||
|  | urlpatterns = [ | ||||||
|  |     # No kwargs | ||||||
|  |     url(r'^conflict/cannot-go-here/$', empty_view, name='name-conflict'), | ||||||
|  |     url(r'^conflict/$', empty_view, name='name-conflict'), | ||||||
|  |     # One kwarg | ||||||
|  |     url(r'^conflict-first/(?P<first>\w+)/$', empty_view, name='name-conflict'), | ||||||
|  |     url(r'^conflict-cannot-go-here/(?P<middle>\w+)/$', empty_view, name='name-conflict'), | ||||||
|  |     url(r'^conflict-middle/(?P<middle>\w+)/$', empty_view, name='name-conflict'), | ||||||
|  |     url(r'^conflict-last/(?P<last>\w+)/$', empty_view, name='name-conflict'), | ||||||
|  |     # Two kwargs | ||||||
|  |     url(r'^conflict/(?P<another>\w+)/(?P<extra>\w+)/cannot-go-here/$', empty_view, name='name-conflict'), | ||||||
|  |     url(r'^conflict/(?P<extra>\w+)/(?P<another>\w+)/$', empty_view, name='name-conflict'), | ||||||
|  | ] | ||||||
| @@ -382,6 +382,23 @@ class ResolverTests(SimpleTestCase): | |||||||
|         self.assertEqual(resolver.reverse('named-url2', 'arg'), 'extra/arg/') |         self.assertEqual(resolver.reverse('named-url2', 'arg'), 'extra/arg/') | ||||||
|         self.assertEqual(resolver.reverse('named-url2', extra='arg'), 'extra/arg/') |         self.assertEqual(resolver.reverse('named-url2', extra='arg'), 'extra/arg/') | ||||||
|  |  | ||||||
|  |     def test_resolver_reverse_conflict(self): | ||||||
|  |         """ | ||||||
|  |         url() name arguments don't need to be unique. The last registered | ||||||
|  |         pattern takes precedence for conflicting names. | ||||||
|  |         """ | ||||||
|  |         resolver = get_resolver('urlpatterns_reverse.named_urls_conflict') | ||||||
|  |         # Without arguments, the last URL in urlpatterns has precedence. | ||||||
|  |         self.assertEqual(resolver.reverse('name-conflict'), 'conflict/') | ||||||
|  |         # With an arg, the last URL in urlpatterns has precedence. | ||||||
|  |         self.assertEqual(resolver.reverse('name-conflict', 'arg'), 'conflict-last/arg/') | ||||||
|  |         # With a kwarg, other url()s can be reversed. | ||||||
|  |         self.assertEqual(resolver.reverse('name-conflict', first='arg'), 'conflict-first/arg/') | ||||||
|  |         self.assertEqual(resolver.reverse('name-conflict', middle='arg'), 'conflict-middle/arg/') | ||||||
|  |         self.assertEqual(resolver.reverse('name-conflict', last='arg'), 'conflict-last/arg/') | ||||||
|  |         # The number and order of the arguments don't interfere with reversing. | ||||||
|  |         self.assertEqual(resolver.reverse('name-conflict', 'arg', 'arg'), 'conflict/arg/arg/') | ||||||
|  |  | ||||||
|     def test_non_regex(self): |     def test_non_regex(self): | ||||||
|         """ |         """ | ||||||
|         A Resolver404 is raised if resolving doesn't meet the basic |         A Resolver404 is raised if resolving doesn't meet the basic | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user