mirror of
				https://github.com/django/django.git
				synced 2025-10-25 06:36:07 +00:00 
			
		
		
		
	Thanks Daniele Procida for review.
This commit is contained in:
		| @@ -2519,6 +2519,7 @@ own ``AdminSite`` instance since you will likely be importing all the per-app | |||||||
| put ``'django.contrib.admin.apps.SimpleAdminConfig'`` instead of | put ``'django.contrib.admin.apps.SimpleAdminConfig'`` instead of | ||||||
| ``'django.contrib.admin'`` in your :setting:`INSTALLED_APPS` setting. | ``'django.contrib.admin'`` in your :setting:`INSTALLED_APPS` setting. | ||||||
|  |  | ||||||
|  | .. _multiple-admin-sites: | ||||||
|  |  | ||||||
| Multiple admin sites in the same URLconf | Multiple admin sites in the same URLconf | ||||||
| ---------------------------------------- | ---------------------------------------- | ||||||
|   | |||||||
| @@ -578,11 +578,21 @@ URL namespaces | |||||||
| Introduction | Introduction | ||||||
| ------------ | ------------ | ||||||
|  |  | ||||||
| When you need to deploy multiple instances of a single application, it can be | URL namespaces allow you to uniquely reverse :ref:`named URL patterns | ||||||
| helpful to be able to differentiate between instances. This is especially | <naming-url-patterns>` even if different applications use the same URL names. | ||||||
| important when using :ref:`named URL patterns <naming-url-patterns>`, since | It's a good practice for third-party apps to always use namespaced URLs (as we | ||||||
| multiple instances of a single application will share named URLs. Namespaces | did in the tutorial). Similarly, it also allows you to reverse URLs if multiple | ||||||
| provide a way to tell these named URLs apart. | instances of an application are deployed. In other words, since multiple | ||||||
|  | instances of a single application will share named URLs, namespaces provide a | ||||||
|  | way to tell these named URLs apart. | ||||||
|  |  | ||||||
|  | Django applications that make proper use of URL namespacing can be deployed more | ||||||
|  | than once for a particular site. For example :mod:`django.contrib.admin` has an | ||||||
|  | :class:`~django.contrib.admin.AdminSite` class which allows you to easily | ||||||
|  | :ref:`deploy more than once instance of the admin <multiple-admin-sites>`. | ||||||
|  | In a later example, we'll discuss the idea of deploying the polls application | ||||||
|  | from the tutorial in two different locations so we can serve the same | ||||||
|  | functionality to two different audiences (authors and publishers). | ||||||
|  |  | ||||||
| A URL namespace comes in two parts, both of which are strings: | A URL namespace comes in two parts, both of which are strings: | ||||||
|  |  | ||||||
| @@ -598,44 +608,43 @@ A URL namespace comes in two parts, both of which are strings: | |||||||
|     This identifies a specific instance of an application. Instance namespaces |     This identifies a specific instance of an application. Instance namespaces | ||||||
|     should be unique across your entire project. However, an instance namespace |     should be unique across your entire project. However, an instance namespace | ||||||
|     can be the same as the application namespace. This is used to specify a |     can be the same as the application namespace. This is used to specify a | ||||||
|     default instance of an application. For example, the default Django Admin |     default instance of an application. For example, the default Django admin | ||||||
|     instance has an instance namespace of ``'admin'``. |     instance has an instance namespace of ``'admin'``. | ||||||
|  |  | ||||||
| Namespaced URLs are specified using the ``':'`` operator. For example, the main | Namespaced URLs are specified using the ``':'`` operator. For example, the main | ||||||
| index page of the admin application is referenced using ``'admin:index'``. This | index page of the admin application is referenced using ``'admin:index'``. This | ||||||
| indicates a namespace of ``'admin'``, and a named URL of ``'index'``. | indicates a namespace of ``'admin'``, and a named URL of ``'index'``. | ||||||
|  |  | ||||||
| Namespaces can also be nested. The named URL ``'foo:bar:whiz'`` would look for | Namespaces can also be nested. The named URL ``'sports:polls:index'`` would | ||||||
| a pattern named ``'whiz'`` in the namespace ``'bar'`` that is itself defined | look for a pattern named ``'index'`` in the namespace ``'polls'`` that is itself | ||||||
| within the top-level namespace ``'foo'``. | defined within the top-level namespace ``'sports'``. | ||||||
|  |  | ||||||
| .. _topics-http-reversing-url-namespaces: | .. _topics-http-reversing-url-namespaces: | ||||||
|  |  | ||||||
| Reversing namespaced URLs | Reversing namespaced URLs | ||||||
| ------------------------- | ------------------------- | ||||||
|  |  | ||||||
| When given a namespaced URL (e.g. ``'myapp:index'``) to resolve, Django splits | When given a namespaced URL (e.g. ``'polls:index'``) to resolve, Django splits | ||||||
| the fully qualified name into parts, and then tries the following lookup: | the fully qualified name into parts and then tries the following lookup: | ||||||
|  |  | ||||||
| 1. First, Django looks for a matching :term:`application namespace` (in this | 1. First, Django looks for a matching :term:`application namespace` (in this | ||||||
|    example, ``'myapp'``). This will yield a list of instances of that |    example, ``'polls'``). This will yield a list of instances of that | ||||||
|    application. |    application. | ||||||
|  |  | ||||||
| 2. If there is a *current* application defined, Django finds and returns | 2. If there is a *current* application defined, Django finds and returns | ||||||
|    the URL resolver for that instance. The *current* application can be |    the URL resolver for that instance. The *current* application can be | ||||||
|    specified as an attribute on the template context - applications that |    specified as an attribute on the template context - applications that | ||||||
|    expect to have multiple deployments should set the ``current_app`` |    expect to have multiple deployments should set the ``current_app`` | ||||||
|    attribute on any ``Context`` or ``RequestContext`` that is used to |    attribute on any :class:`~django.template.Context` or | ||||||
|    render a template. |    :class:`~django.template.RequestContext` that is used to render a template. | ||||||
|  |  | ||||||
|    The current application can also be specified manually as an argument |    The current application can also be specified manually as an argument | ||||||
|    to the :func:`django.core.urlresolvers.reverse` function. |    to the :func:`~django.core.urlresolvers.reverse` function. | ||||||
|  |  | ||||||
| 3. If there is no current application. Django looks for a default | 3. If there is no current application. Django looks for a default | ||||||
|    application instance. The default application instance is the instance |    application instance. The default application instance is the instance | ||||||
|    that has an :term:`instance namespace` matching the :term:`application |    that has an :term:`instance namespace` matching the :term:`application | ||||||
|    namespace` (in this example, an instance of the ``myapp`` called |    namespace` (in this example, an instance of ``polls`` called ``'polls'``). | ||||||
|    ``'myapp'``). |  | ||||||
|  |  | ||||||
| 4. If there is no default application instance, Django will pick the last | 4. If there is no default application instance, Django will pick the last | ||||||
|    deployed instance of the application, whatever its instance name may be. |    deployed instance of the application, whatever its instance name may be. | ||||||
| @@ -652,37 +661,73 @@ Example | |||||||
| ~~~~~~~ | ~~~~~~~ | ||||||
|  |  | ||||||
| To show this resolution strategy in action, consider an example of two instances | To show this resolution strategy in action, consider an example of two instances | ||||||
| of ``myapp``: one called ``'foo'``, and one called ``'bar'``. ``myapp`` has a | of the ``polls`` application from the tutorial: one called ``'author-polls'`` | ||||||
| main index page with a URL named ``'index'``. Using this setup, the following | and one called ``'publisher-polls'``. Assume we have enhanced that application | ||||||
| lookups are possible: | so that it takes the instance namespace into consideration when creating and | ||||||
|  | displaying polls. | ||||||
|  |  | ||||||
| * If one of the instances is current - say, if we were rendering a utility page | .. snippet:: | ||||||
|   in the instance ``'bar'`` - ``'myapp:index'`` will resolve to the index page |     :filename: urls.py | ||||||
|   of the instance ``'bar'``. |  | ||||||
|  |     from django.conf.urls import include, url | ||||||
|  |  | ||||||
|  |     urlpatterns = [ | ||||||
|  |         url(r'^author-polls/', include('polls.urls', namespace='author-polls', app_name='polls')), | ||||||
|  |         url(r'^publisher-polls/', include('polls.urls', namespace='publisher-polls', app_name='polls')), | ||||||
|  |     ] | ||||||
|  |  | ||||||
|  | .. snippet:: | ||||||
|  |     :filename: polls/urls.py | ||||||
|  |  | ||||||
|  |     from django.conf.urls import url | ||||||
|  |  | ||||||
|  |     from . import views | ||||||
|  |  | ||||||
|  |     urlpatterns = [ | ||||||
|  |         url(r'^$', views.IndexView.as_view(), name='index'), | ||||||
|  |         url(r'^(?P<pk>\d+)/$', views.DetailView.as_view(), name='detail'), | ||||||
|  |         ... | ||||||
|  |     ] | ||||||
|  |  | ||||||
|  | Using this setup, the following lookups are possible: | ||||||
|  |  | ||||||
|  | * If one of the instances is current - say, if we were rendering the detail page | ||||||
|  |   in the instance ``'author-polls'`` - ``'polls:index'`` will resolve to the | ||||||
|  |   index page of the ``'author-polls'`` instance; i.e. both of the following will | ||||||
|  |   result in ``"/author-polls/"``. | ||||||
|  |  | ||||||
|  |   In the method of a class-based view:: | ||||||
|  |  | ||||||
|  |     reverse('polls:index', current_app=self.request.resolver_match.namespace) | ||||||
|  |  | ||||||
|  |   and in the template: | ||||||
|  |  | ||||||
|  |   .. code-block:: html+django | ||||||
|  |  | ||||||
|  |     {% url 'polls:index' %} | ||||||
|  |  | ||||||
|  |   Note that reversing in the template requires the ``current_app`` be added as | ||||||
|  |   an attribute to the template context like this:: | ||||||
|  |  | ||||||
|  |     def render_to_response(self, context, **response_kwargs): | ||||||
|  |         response_kwargs['current_app'] = self.request.resolver_match.namespace | ||||||
|  |         return super(DetailView, self).render_to_response(context, **response_kwargs) | ||||||
|  |  | ||||||
| * If there is no current instance - say, if we were rendering a page | * If there is no current instance - say, if we were rendering a page | ||||||
|   somewhere else on the site - ``'myapp:index'`` will resolve to the last |   somewhere else on the site - ``'polls:index'`` will resolve to the last | ||||||
|   registered instance of ``myapp``. Since there is no default instance, |   registered instance of ``polls``. Since there is no default instance | ||||||
|   the last instance of ``myapp`` that is registered will be used. This could |   (instance namespace of ``'polls'``), the last instance of ``polls`` that is | ||||||
|   be ``'foo'`` or ``'bar'``, depending on the order they are introduced into the |   registered will be used. This would be ``'publisher-polls'`` since it's | ||||||
|   urlpatterns of the project. |   declared last in the ``urlpatterns``. | ||||||
|  |  | ||||||
| * ``'foo:index'`` will always resolve to the index page of the instance | * ``'author-polls:index'`` will always resolve to the index page of the instance | ||||||
|   ``'foo'``. |   ``'author-polls'`` (and likewise for ``'publisher-polls'``) . | ||||||
|  |  | ||||||
| If there was also a default instance - i.e., an instance named ``'myapp'`` - the | If there were also a default instance - i.e., an instance named ``'polls'`` - | ||||||
| following would happen: | the only change from above would be in the case where there is no current | ||||||
|  | instance (the second item in the list above). In this case ``'polls:index'`` | ||||||
| * If one of the instances is current - say, if we were rendering a utility page | would resolve to the index page of the default instance instead of the instance | ||||||
|   in the instance ``'bar'`` - ``'myapp:index'`` will resolve to the index page | declared last in ``urlpatterns``. | ||||||
|   of the instance ``'bar'``. |  | ||||||
|  |  | ||||||
| * If there is no current instance - say, if we were rendering a page somewhere |  | ||||||
|   else on the site - ``'myapp:index'`` will resolve to the index page of the |  | ||||||
|   default instance. |  | ||||||
|  |  | ||||||
| * ``'foo:index'`` will again resolve to the index page of the instance |  | ||||||
|   ``'foo'``. |  | ||||||
|  |  | ||||||
| .. _namespaces-and-include: | .. _namespaces-and-include: | ||||||
|  |  | ||||||
| @@ -693,17 +738,17 @@ URL namespaces of included URLconfs can be specified in two ways. | |||||||
|  |  | ||||||
| Firstly, you can provide the :term:`application <application namespace>` and | Firstly, you can provide the :term:`application <application namespace>` and | ||||||
| :term:`instance <instance namespace>` namespaces as arguments to | :term:`instance <instance namespace>` namespaces as arguments to | ||||||
| :func:`django.conf.urls.include()` when you construct your URL patterns. For | :func:`~django.conf.urls.include()` when you construct your URL patterns. For | ||||||
| example,:: | example,:: | ||||||
|  |  | ||||||
|     url(r'^help/', include('apps.help.urls', namespace='foo', app_name='bar')), |     url(r'^polls/', include('polls.urls', namespace='author-polls', app_name='polls')), | ||||||
|  |  | ||||||
| This will include the URLs defined in ``apps.help.urls`` into the | This will include the URLs defined in ``polls.urls`` into the | ||||||
| :term:`application namespace` ``'bar'``, with the :term:`instance namespace` | :term:`application namespace` ``'polls'``, with the :term:`instance namespace` | ||||||
| ``'foo'``. | ``'author-polls'``. | ||||||
|  |  | ||||||
| Secondly, you can include an object that contains embedded namespace data. If | Secondly, you can include an object that contains embedded namespace data. If | ||||||
| you ``include()`` a list of :func:`django.conf.urls.url` instances, | you ``include()`` a list of :func:`~django.conf.urls.url` instances, | ||||||
| the URLs contained in that object will be added to the global namespace. | the URLs contained in that object will be added to the global namespace. | ||||||
| However, you can also ``include()`` a 3-tuple containing:: | However, you can also ``include()`` a 3-tuple containing:: | ||||||
|  |  | ||||||
| @@ -713,26 +758,27 @@ For example:: | |||||||
|  |  | ||||||
|     from django.conf.urls import include, url |     from django.conf.urls import include, url | ||||||
|  |  | ||||||
|     from app.helps import views |     from . import views | ||||||
|  |  | ||||||
|     help_patterns = [ |     polls_patterns = [ | ||||||
|         url(r'^basic/$', views.basic), |         url(r'^$', views.IndexView.as_view(), name='index'), | ||||||
|         url(r'^advanced/$', views.advanced), |         url(r'^(?P<pk>\d+)/$', views.DetailView.as_view(), name='detail'), | ||||||
|     ] |     ] | ||||||
|  |  | ||||||
|     url(r'^help/', include((help_patterns, 'bar', 'foo'))), |     url(r'^polls/', include((polls_patterns, 'polls', 'author-polls'))), | ||||||
|  |  | ||||||
| This will include the nominated URL patterns into the given application and | This will include the nominated URL patterns into the given application and | ||||||
| instance namespace. | instance namespace. | ||||||
|  |  | ||||||
| For example, the Django Admin is deployed as instances of | For example, the Django admin is deployed as instances of | ||||||
| :class:`~django.contrib.admin.AdminSite`.  ``AdminSite`` objects have a ``urls`` | :class:`~django.contrib.admin.AdminSite`.  ``AdminSite`` objects have a ``urls`` | ||||||
| attribute: A 3-tuple that contains all the patterns in the corresponding admin | attribute: A 3-tuple that contains all the patterns in the corresponding admin | ||||||
| site, plus the application namespace ``'admin'``, and the name of the admin | site, plus the application namespace ``'admin'``, and the name of the admin | ||||||
| instance. It is this ``urls`` attribute that you ``include()`` into your | instance. It is this ``urls`` attribute that you ``include()`` into your | ||||||
| projects ``urlpatterns`` when you deploy an Admin instance. | projects ``urlpatterns`` when you deploy an admin instance. | ||||||
|  |  | ||||||
| Be sure to pass a tuple to ``include()``. If you simply pass three arguments: | Be sure to pass a tuple to ``include()``. If you simply pass three arguments: | ||||||
| ``include(help_patterns, 'bar', 'foo')``, Django won't throw an error but due | ``include(polls_patterns, 'polls', 'author-polls')``, Django won't throw an | ||||||
| to the signature of ``include()``, ``'bar'`` will be the instance namespace and | error but due to the signature of ``include()``, ``'polls'`` will be the | ||||||
| ``'foo'`` will be the application namespace instead of vice versa. | instance namespace and ``'author-polls'`` will be the application namespace | ||||||
|  | instead of vice versa. | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user