mirror of
				https://github.com/django/django.git
				synced 2025-10-25 14:46:09 +00:00 
			
		
		
		
	Added docs/sites.txt -- documentation on the sites framework
git-svn-id: http://code.djangoproject.com/svn/django/trunk@2958 bcc190cf-cafb-0310-a4f2-bffc1f526a37
This commit is contained in:
		| @@ -77,7 +77,11 @@ sites | |||||||
|  |  | ||||||
| A light framework that lets you operate multiple Web sites off of the same | A light framework that lets you operate multiple Web sites off of the same | ||||||
| database and Django installation. It gives you hooks for associating objects to | database and Django installation. It gives you hooks for associating objects to | ||||||
| one or more sites. This is not yet documented. | one or more sites. | ||||||
|  |  | ||||||
|  | See the `sites documentation`_. | ||||||
|  |  | ||||||
|  | .. _sites documentation: http://www.djangoproject.com/documentation/sites/ | ||||||
|  |  | ||||||
| syndication | syndication | ||||||
| =========== | =========== | ||||||
|   | |||||||
| @@ -355,7 +355,7 @@ variables: | |||||||
|     * ``next``: The URL to redirect to after successful login. This may contain |     * ``next``: The URL to redirect to after successful login. This may contain | ||||||
|       a query string, too. |       a query string, too. | ||||||
|     * ``site_name``: The name of the current ``Site``, according to the |     * ``site_name``: The name of the current ``Site``, according to the | ||||||
|       ``SITE_ID`` setting. |       ``SITE_ID`` setting. See the `site framework docs`_. | ||||||
|  |  | ||||||
| Here's a sample ``registration/login.html`` template you can use as a starting | Here's a sample ``registration/login.html`` template you can use as a starting | ||||||
| point. It assumes you have a ``base.html`` template that defines a ``content`` | point. It assumes you have a ``base.html`` template that defines a ``content`` | ||||||
| @@ -382,6 +382,7 @@ block:: | |||||||
|     {% endblock %} |     {% endblock %} | ||||||
|  |  | ||||||
| .. _forms documentation: http://www.djangoproject.com/documentation/forms/ | .. _forms documentation: http://www.djangoproject.com/documentation/forms/ | ||||||
|  | .. _site framework docs: http://www.djangoproject.com/documentation/sites/ | ||||||
|  |  | ||||||
| Limiting access to logged-in users that pass a test | Limiting access to logged-in users that pass a test | ||||||
| --------------------------------------------------- | --------------------------------------------------- | ||||||
|   | |||||||
| @@ -619,6 +619,10 @@ The ID, as an integer, of the current site in the ``django_site`` database | |||||||
| table. This is used so that application data can hook into specific site(s) | table. This is used so that application data can hook into specific site(s) | ||||||
| and a single database can manage content for multiple sites. | and a single database can manage content for multiple sites. | ||||||
|  |  | ||||||
|  | See the `site framework docs`_. | ||||||
|  |  | ||||||
|  | .. _site framework docs: http://www.djangoproject.com/documentation/sites/ | ||||||
|  |  | ||||||
| TEMPLATE_CONTEXT_PROCESSORS | TEMPLATE_CONTEXT_PROCESSORS | ||||||
| --------------------------- | --------------------------- | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										256
									
								
								docs/sites.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										256
									
								
								docs/sites.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,256 @@ | |||||||
|  | ===================== | ||||||
|  | The "sites" framework | ||||||
|  | ===================== | ||||||
|  |  | ||||||
|  | Django comes with an optional "sites" framework. It's a hook for associating | ||||||
|  | objects and functionality to particular Web sites, and it's a holding place for | ||||||
|  | the domain names and "verbose" names of your Django-powered sites. | ||||||
|  |  | ||||||
|  | Use it if your single Django installation powers more than one site and you | ||||||
|  | need to differentiate between those sites in some way. | ||||||
|  |  | ||||||
|  | The whole sites framework is based on two simple concepts: | ||||||
|  |  | ||||||
|  |     * The ``Site`` model, found in ``django.contrib.sites``, has ``domain`` and | ||||||
|  |       ``name`` fields. | ||||||
|  |     * The ``SITE_ID`` setting specifies the database ID of the ``Site`` object | ||||||
|  |       associated with that particular settings file. | ||||||
|  |  | ||||||
|  | How you use this is up to you, but Django uses it in a couple of ways | ||||||
|  | automatically via simple conventions. | ||||||
|  |  | ||||||
|  | Example usage | ||||||
|  | ============= | ||||||
|  |  | ||||||
|  | Why would you use sites? It's best explained through examples. | ||||||
|  |  | ||||||
|  | Associating content with multiple sites | ||||||
|  | --------------------------------------- | ||||||
|  |  | ||||||
|  | The Django-powered sites LJWorld.com_ and Lawrence.com_ are operated by the | ||||||
|  | same news organization -- the Lawrence Journal-World newspaper in Lawrence, | ||||||
|  | Kansas. LJWorld.com focuses on news, while Lawrence.com focuses on local | ||||||
|  | entertainment. But sometimes editors want to publish an article on *both* | ||||||
|  | sites. | ||||||
|  |  | ||||||
|  | The brain-dead way of solving the problem would be to require site producers to | ||||||
|  | publish the same story twice: once for LJWorld.com and again for Lawrence.com. | ||||||
|  | But that's inefficient for site producers, and it's redundant to store | ||||||
|  | multiple copies of the same story in the database. | ||||||
|  |  | ||||||
|  | The better solution is simple: Both sites use the same article database, and an | ||||||
|  | article is associated with one or more sites. In Django model terminology, | ||||||
|  | that's represented by a ``ManyToManyField`` in the ``Article`` model:: | ||||||
|  |  | ||||||
|  |     from django.db import models | ||||||
|  |     from django.contrib.sites.models import Site | ||||||
|  |  | ||||||
|  |     class Article(models.Model): | ||||||
|  |         headline = models.CharField(maxlength=200) | ||||||
|  |         # ... | ||||||
|  |         sites = models.ManyToManyField(Site) | ||||||
|  |  | ||||||
|  | This accomplishes several things quite nicely: | ||||||
|  |  | ||||||
|  |     * It lets the site producers edit all content -- on both sites -- in a | ||||||
|  |       single interface (the Django admin). | ||||||
|  |  | ||||||
|  |     * It means the same story doesn't have to be published twice in the | ||||||
|  |       database; it only has a single record in the database. | ||||||
|  |  | ||||||
|  |     * It lets the site developers use the same Django view code for both sites. | ||||||
|  |       The view code that displays a given story just checks to make sure the | ||||||
|  |       requested story is on the current site. It looks something like this:: | ||||||
|  |  | ||||||
|  |           from django.conf import settings | ||||||
|  |  | ||||||
|  |           def article_detail(request, article_id): | ||||||
|  |               try: | ||||||
|  |                   a = Article.objects.get(id=article_id, sites__id__exact=settings.SITE_ID) | ||||||
|  |               except Article.DoesNotExist: | ||||||
|  |                   raise Http404 | ||||||
|  |               # ... | ||||||
|  |  | ||||||
|  | .. _ljworld.com: http://www.ljworld.com/ | ||||||
|  | .. _lawrence.com: http://www.lawrence.com/ | ||||||
|  |  | ||||||
|  | Associating content with a single site | ||||||
|  | -------------------------------------- | ||||||
|  |  | ||||||
|  | Similarly, you can associate a model to the ``Site`` model in a many-to-one | ||||||
|  | relationship, using ``ForeignKey``. | ||||||
|  |  | ||||||
|  | For example, if an article is only allowed on a single site, you'd use a model | ||||||
|  | like this:: | ||||||
|  |  | ||||||
|  |     from django.db import models | ||||||
|  |     from django.contrib.sites.models import Site | ||||||
|  |  | ||||||
|  |     class Article(models.Model): | ||||||
|  |         headline = models.CharField(maxlength=200) | ||||||
|  |         # ... | ||||||
|  |         site = models.ForeignKey(Site) | ||||||
|  |  | ||||||
|  | This has the same benefits as described in the last section. | ||||||
|  |  | ||||||
|  | Hooking into the current site from views | ||||||
|  | ---------------------------------------- | ||||||
|  |  | ||||||
|  | On a lower level, you can use the sites framework in your Django views to do | ||||||
|  | particular things based on what site in which the view is being called. | ||||||
|  | For example:: | ||||||
|  |  | ||||||
|  |     from django.conf import settings | ||||||
|  |  | ||||||
|  |     def my_view(request): | ||||||
|  |         if settings.SITE_ID == 3: | ||||||
|  |             # Do something. | ||||||
|  |         else: | ||||||
|  |             # Do something else. | ||||||
|  |  | ||||||
|  | Of course, it's ugly to hard-code the site IDs like that. This sort of | ||||||
|  | hard-coding is best for hackish fixes that you need done quickly. A slightly | ||||||
|  | cleaner way of accomplishing the same thing is to check the current site's | ||||||
|  | domain:: | ||||||
|  |  | ||||||
|  |     from django.conf import settings | ||||||
|  |     from django.contrib.sites.models import Site | ||||||
|  |  | ||||||
|  |     def my_view(request): | ||||||
|  |         current_site = Site.objects.get(id=settings.SITE_ID) | ||||||
|  |         if current_site.domain == 'foo.com': | ||||||
|  |             # Do something | ||||||
|  |         else: | ||||||
|  |             # Do something else. | ||||||
|  |  | ||||||
|  | The idiom of retrieving the ``Site`` object for the value of | ||||||
|  | ``settings.SITE_ID`` is quite common, so the ``Site`` model's manager has a | ||||||
|  | ``get_current()`` method. This example is equivalent to the previous one:: | ||||||
|  |  | ||||||
|  |     from django.contrib.sites.models import Site | ||||||
|  |  | ||||||
|  |     def my_view(request): | ||||||
|  |         current_site = Site.objects.get_current() | ||||||
|  |         if current_site.domain == 'foo.com': | ||||||
|  |             # Do something | ||||||
|  |         else: | ||||||
|  |             # Do something else. | ||||||
|  |  | ||||||
|  | Getting the current domain for display | ||||||
|  | -------------------------------------- | ||||||
|  |  | ||||||
|  | LJWorld.com and Lawrence.com both have e-mail alert functionality, which lets | ||||||
|  | readers sign up to get notifications when news happens. It's pretty basic: A | ||||||
|  | reader signs up on a Web form, and he immediately gets an e-mail saying, | ||||||
|  | "Thanks for your subscription." | ||||||
|  |  | ||||||
|  | It'd be inefficient and redundant to implement this signup-processing code | ||||||
|  | twice, so the sites use the same code behind the scenes. But the "thank you for | ||||||
|  | signing up" notice needs to be different for each site. By using ``Site`` | ||||||
|  | objects, we can abstract the "thank you" notice to use the values of the | ||||||
|  | current site's ``name`` and ``domain``. | ||||||
|  |  | ||||||
|  | Here's an example of what the form-handling view looks like:: | ||||||
|  |  | ||||||
|  |     from django.contrib.sites.models import Site | ||||||
|  |     from django.core.mail import send_mail | ||||||
|  |  | ||||||
|  |     def register_for_newsletter(request): | ||||||
|  |         # Check form values, etc., and subscribe the user. | ||||||
|  |         # ... | ||||||
|  |  | ||||||
|  |         current_site = Site.objects.get_current() | ||||||
|  |         send_mail('Thanks for subscribing to %s alerts' % current_site.name, | ||||||
|  |             'Thanks for your subscription. We appreciate it.\n\n-The %s team.' % current_site.name, | ||||||
|  |             'editor@%s' % current_site.domain, | ||||||
|  |             [user.email]) | ||||||
|  |  | ||||||
|  |         # ... | ||||||
|  |  | ||||||
|  | On Lawrence.com, this e-mail has the subject line "Thanks for subscribing to | ||||||
|  | lawrence.com alerts." On LJWorld.com, the e-mail has the subject "Thanks for | ||||||
|  | subscribing to LJWorld.com alerts." Same goes for the e-mail's message body. | ||||||
|  |  | ||||||
|  | Note that an even more flexible (but more heavyweight) way of doing this would | ||||||
|  | be to use Django's template system. Assuming Lawrence.com and LJWorld.com have | ||||||
|  | different template directories (``TEMPLATE_DIRS``), you could simply farm out | ||||||
|  | to the template system like so:: | ||||||
|  |  | ||||||
|  |     from django.core.mail import send_mail | ||||||
|  |     from django.template import loader, Context | ||||||
|  |  | ||||||
|  |     def register_for_newsletter(request): | ||||||
|  |         # Check form values, etc., and subscribe the user. | ||||||
|  |         # ... | ||||||
|  |  | ||||||
|  |         subject = loader.get_template('alerts/subject.txt').render(Context({})) | ||||||
|  |         message = loader.get_template('alerts/message.txt').render(Context({})) | ||||||
|  |         send_mail(subject, message, 'editor@ljworld.com', [user.email]) | ||||||
|  |  | ||||||
|  |         # ... | ||||||
|  |  | ||||||
|  | In this case, you'd have to create ``subject.txt`` and ``message.txt`` template | ||||||
|  | files for both the LJWorld.com and Lawrence.com template directories. That | ||||||
|  | gives you more flexibility, but it's also more complex. | ||||||
|  |  | ||||||
|  | It's a good idea to exploit the ``Site`` objects as much as possible, to remove | ||||||
|  | unneeded complexity and redundancy. | ||||||
|  |  | ||||||
|  | Getting the current domain for full URLs | ||||||
|  | ---------------------------------------- | ||||||
|  |  | ||||||
|  | Django's ``get_absolute_url()`` convention is nice for getting your objects' | ||||||
|  | URL without the domain name, but in some cases you might want to display the | ||||||
|  | full URL -- with ``http://`` and the domain and everything -- for an object. | ||||||
|  | To do this, you can use the sites framework. A simple example:: | ||||||
|  |  | ||||||
|  |     >>> from django.contrib.sites.models import Site | ||||||
|  |     >>> obj = MyModel.objects.get(id=3) | ||||||
|  |     >>> obj.get_absolute_url() | ||||||
|  |     '/mymodel/objects/3/' | ||||||
|  |     >>> Site.objects.get_current().domain | ||||||
|  |     'example.com' | ||||||
|  |     >>> 'http://%s%s' % (Site.objects.get_current().domain, obj.get_absolute_url()) | ||||||
|  |     'http://example.com/mymodel/objects/3/' | ||||||
|  |  | ||||||
|  | How Django uses the sites framework | ||||||
|  | =================================== | ||||||
|  |  | ||||||
|  | Although it's not required that you use the sites framework, it's strongly | ||||||
|  | encouraged, because Django takes advantage of it in a few places. Even if your | ||||||
|  | Django installation is powering only a single site, you should take the two | ||||||
|  | seconds to create the site object with your ``domain`` and ``name``, and point | ||||||
|  | to its ID in your ``SITE_ID`` setting. | ||||||
|  |  | ||||||
|  | Here's how Django uses the sites framework: | ||||||
|  |  | ||||||
|  |     * In the `redirects framework`_, each redirect object is associated with a | ||||||
|  |       particular site. When Django searches for a redirect, it takes into | ||||||
|  |       account the current ``SITE_ID``. | ||||||
|  |  | ||||||
|  |     * In the comments framework, each comment is associated with a particular | ||||||
|  |       site. When a comment is posted, its ``site`` is set to the current | ||||||
|  |       ``SITE_ID``, and when comments are listed via the appropriate template | ||||||
|  |       tag, only the comments for the current site are displayed. | ||||||
|  |  | ||||||
|  |     * In the `flatpages framework`_, each flatpage is associated with a | ||||||
|  |       particular site. When a flatpage is created, you specify its ``site``, | ||||||
|  |       and the ``FlatpageFallbackMiddleware`` checks the current ``SITE_ID`` in | ||||||
|  |       retrieving flatpages to display. | ||||||
|  |  | ||||||
|  |     * In the `syndication framework`_, the templates for ``title`` and | ||||||
|  |       ``description`` automatically have access to a variable ``{{{ site }}}``, | ||||||
|  |       which is the ``Site`` object representing the current site. Also, the | ||||||
|  |       hook for providing item URLs will use the ``domain`` from the current | ||||||
|  |       ``Site`` object if you don't specify a fully-qualified domain. | ||||||
|  |  | ||||||
|  |     * In the `authentication framework`_, the ``django.contrib.auth.views.login`` | ||||||
|  |       view passes the current ``Site`` name to the template as ``{{{ site_name }}}``. | ||||||
|  |  | ||||||
|  |     * The shortcut view (``django.views.defaults.shortcut``) uses the domain of | ||||||
|  |       the current ``Site`` object when calculating an object's URL. | ||||||
|  |  | ||||||
|  | .. _redirects framework: http://www.djangoproject.com/documentation/redirects/ | ||||||
|  | .. _flatpages framework: http://www.djangoproject.com/documentation/flatpages/ | ||||||
|  | .. _syndication framework: http://www.djangoproject.com/documentation/syndication/ | ||||||
|  | .. _authentication framework: http://www.djangoproject.com/documentation/syndication/ | ||||||
		Reference in New Issue
	
	Block a user