sites.txt 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494
  1. =====================
  2. The "sites" framework
  3. =====================
  4. .. module:: django.contrib.sites
  5. :synopsis: Lets you operate multiple Web sites from the same database and
  6. Django project
  7. Django comes with an optional "sites" framework. It's a hook for associating
  8. objects and functionality to particular Web sites, and it's a holding place for
  9. the domain names and "verbose" names of your Django-powered sites.
  10. Use it if your single Django installation powers more than one site and you
  11. need to differentiate between those sites in some way.
  12. The sites framework is mainly based on a simple model:
  13. .. class:: models.Site
  14. A model for storing the ``domain`` and ``name`` attributes of a Web site.
  15. The :setting:`SITE_ID` setting specifies the database ID of the
  16. :class:`~django.contrib.sites.models.Site` object associated with that
  17. particular settings file.
  18. .. attribute:: domain
  19. The domain name associated with the Web site.
  20. .. attribute:: name
  21. A human-readable "verbose" name for the Web site.
  22. How you use this is up to you, but Django uses it in a couple of ways
  23. automatically via simple conventions.
  24. Example usage
  25. =============
  26. Why would you use sites? It's best explained through examples.
  27. Associating content with multiple sites
  28. ---------------------------------------
  29. The Django-powered sites LJWorld.com_ and Lawrence.com_ are operated by the
  30. same news organization -- the Lawrence Journal-World newspaper in Lawrence,
  31. Kansas. LJWorld.com focuses on news, while Lawrence.com focuses on local
  32. entertainment. But sometimes editors want to publish an article on *both*
  33. sites.
  34. The brain-dead way of solving the problem would be to require site producers to
  35. publish the same story twice: once for LJWorld.com and again for Lawrence.com.
  36. But that's inefficient for site producers, and it's redundant to store
  37. multiple copies of the same story in the database.
  38. The better solution is simple: Both sites use the same article database, and an
  39. article is associated with one or more sites. In Django model terminology,
  40. that's represented by a :class:`~django.db.models.ManyToManyField` in the
  41. ``Article`` model::
  42. from django.db import models
  43. from django.contrib.sites.models import Site
  44. class Article(models.Model):
  45. headline = models.CharField(max_length=200)
  46. # ...
  47. sites = models.ManyToManyField(Site)
  48. This accomplishes several things quite nicely:
  49. * It lets the site producers edit all content -- on both sites -- in a
  50. single interface (the Django admin).
  51. * It means the same story doesn't have to be published twice in the
  52. database; it only has a single record in the database.
  53. * It lets the site developers use the same Django view code for both sites.
  54. The view code that displays a given story just checks to make sure the
  55. requested story is on the current site. It looks something like this::
  56. from django.contrib.sites.shortcuts import get_current_site
  57. def article_detail(request, article_id):
  58. try:
  59. a = Article.objects.get(id=article_id, sites__id=get_current_site(request).id)
  60. except Article.DoesNotExist:
  61. raise Http404
  62. # ...
  63. .. _ljworld.com: http://www.ljworld.com/
  64. .. _lawrence.com: http://www.lawrence.com/
  65. Associating content with a single site
  66. --------------------------------------
  67. Similarly, you can associate a model to the
  68. :class:`~django.contrib.sites.models.Site`
  69. model in a many-to-one relationship, using
  70. :class:`~django.db.models.ForeignKey`.
  71. For example, if an article is only allowed on a single site, you'd use a model
  72. like this::
  73. from django.db import models
  74. from django.contrib.sites.models import Site
  75. class Article(models.Model):
  76. headline = models.CharField(max_length=200)
  77. # ...
  78. site = models.ForeignKey(Site)
  79. This has the same benefits as described in the last section.
  80. .. _hooking-into-current-site-from-views:
  81. Hooking into the current site from views
  82. ----------------------------------------
  83. You can use the sites framework in your Django views to do
  84. particular things based on the site in which the view is being called.
  85. For example::
  86. from django.conf import settings
  87. def my_view(request):
  88. if settings.SITE_ID == 3:
  89. # Do something.
  90. pass
  91. else:
  92. # Do something else.
  93. pass
  94. Of course, it's ugly to hard-code the site IDs like that. This sort of
  95. hard-coding is best for hackish fixes that you need done quickly. The
  96. cleaner way of accomplishing the same thing is to check the current site's
  97. domain::
  98. from django.contrib.sites.shortcuts import get_current_site
  99. def my_view(request):
  100. current_site = get_current_site(request)
  101. if current_site.domain == 'foo.com':
  102. # Do something
  103. pass
  104. else:
  105. # Do something else.
  106. pass
  107. This has also the advantage of checking if the sites framework is installed,
  108. and return a :class:`~django.contrib.sites.requests.RequestSite` instance if
  109. it is not.
  110. If you don't have access to the request object, you can use the
  111. ``get_current()`` method of the :class:`~django.contrib.sites.models.Site`
  112. model's manager. You should then ensure that your settings file does contain
  113. the :setting:`SITE_ID` setting. This example is equivalent to the previous one::
  114. from django.contrib.sites.models import Site
  115. def my_function_without_request():
  116. current_site = Site.objects.get_current()
  117. if current_site.domain == 'foo.com':
  118. # Do something
  119. pass
  120. else:
  121. # Do something else.
  122. pass
  123. Getting the current domain for display
  124. --------------------------------------
  125. LJWorld.com and Lawrence.com both have email alert functionality, which lets
  126. readers sign up to get notifications when news happens. It's pretty basic: A
  127. reader signs up on a Web form and immediately gets an email saying,
  128. "Thanks for your subscription."
  129. It'd be inefficient and redundant to implement this sign up processing code
  130. twice, so the sites use the same code behind the scenes. But the "thank you for
  131. signing up" notice needs to be different for each site. By using
  132. :class:`~django.contrib.sites.models.Site`
  133. objects, we can abstract the "thank you" notice to use the values of the
  134. current site's :attr:`~django.contrib.sites.models.Site.name` and
  135. :attr:`~django.contrib.sites.models.Site.domain`.
  136. Here's an example of what the form-handling view looks like::
  137. from django.contrib.sites.shortcuts import get_current_site
  138. from django.core.mail import send_mail
  139. def register_for_newsletter(request):
  140. # Check form values, etc., and subscribe the user.
  141. # ...
  142. current_site = get_current_site(request)
  143. send_mail('Thanks for subscribing to %s alerts' % current_site.name,
  144. 'Thanks for your subscription. We appreciate it.\n\n-The %s team.' % current_site.name,
  145. 'editor@%s' % current_site.domain,
  146. [user.email])
  147. # ...
  148. On Lawrence.com, this email has the subject line "Thanks for subscribing to
  149. lawrence.com alerts." On LJWorld.com, the email has the subject "Thanks for
  150. subscribing to LJWorld.com alerts." Same goes for the email's message body.
  151. Note that an even more flexible (but more heavyweight) way of doing this would
  152. be to use Django's template system. Assuming Lawrence.com and LJWorld.com have
  153. different template directories (:setting:`TEMPLATE_DIRS`), you could simply
  154. farm out to the template system like so::
  155. from django.core.mail import send_mail
  156. from django.template import loader, Context
  157. def register_for_newsletter(request):
  158. # Check form values, etc., and subscribe the user.
  159. # ...
  160. subject = loader.get_template('alerts/subject.txt').render(Context({}))
  161. message = loader.get_template('alerts/message.txt').render(Context({}))
  162. send_mail(subject, message, 'editor@ljworld.com', [user.email])
  163. # ...
  164. In this case, you'd have to create :file:`subject.txt` and :file:`message.txt`
  165. template files for both the LJWorld.com and Lawrence.com template directories.
  166. That gives you more flexibility, but it's also more complex.
  167. It's a good idea to exploit the :class:`~django.contrib.sites.models.Site`
  168. objects as much as possible, to remove unneeded complexity and redundancy.
  169. Getting the current domain for full URLs
  170. ----------------------------------------
  171. Django's ``get_absolute_url()`` convention is nice for getting your objects'
  172. URL without the domain name, but in some cases you might want to display the
  173. full URL -- with ``http://`` and the domain and everything -- for an object.
  174. To do this, you can use the sites framework. A simple example::
  175. >>> from django.contrib.sites.models import Site
  176. >>> obj = MyModel.objects.get(id=3)
  177. >>> obj.get_absolute_url()
  178. '/mymodel/objects/3/'
  179. >>> Site.objects.get_current().domain
  180. 'example.com'
  181. >>> 'http://%s%s' % (Site.objects.get_current().domain, obj.get_absolute_url())
  182. 'http://example.com/mymodel/objects/3/'
  183. .. _enabling-the-sites-framework:
  184. Enabling the sites framework
  185. ============================
  186. To enable the sites framework, follow these steps:
  187. 1. Add ``'django.contrib.sites'`` to your :setting:`INSTALLED_APPS`
  188. setting.
  189. 2. Define a :setting:`SITE_ID` setting::
  190. SITE_ID = 1
  191. 3. Run :djadmin:`migrate`.
  192. ``django.contrib.sites`` registers a
  193. :data:`~django.db.models.signals.post_migrate` signal handler which creates a
  194. default site named ``example.com`` with the domain ``example.com``. This site
  195. will also be created after Django creates the test database. To set the
  196. correct name and domain for your project, you can use an :doc:`initial data
  197. fixture </howto/initial-data>`.
  198. Caching the current ``Site`` object
  199. ===================================
  200. As the current site is stored in the database, each call to
  201. ``Site.objects.get_current()`` could result in a database query. But Django is a
  202. little cleverer than that: on the first request, the current site is cached, and
  203. any subsequent call returns the cached data instead of hitting the database.
  204. If for any reason you want to force a database query, you can tell Django to
  205. clear the cache using ``Site.objects.clear_cache()``::
  206. # First call; current site fetched from database.
  207. current_site = Site.objects.get_current()
  208. # ...
  209. # Second call; current site fetched from cache.
  210. current_site = Site.objects.get_current()
  211. # ...
  212. # Force a database query for the third call.
  213. Site.objects.clear_cache()
  214. current_site = Site.objects.get_current()
  215. The ``CurrentSiteManager``
  216. ==========================
  217. .. class:: managers.CurrentSiteManager
  218. If :class:`~django.contrib.sites.models.Site` plays a key role in your
  219. application, consider using the helpful
  220. :class:`~django.contrib.sites.managers.CurrentSiteManager` in your
  221. model(s). It's a model :doc:`manager </topics/db/managers>` that
  222. automatically filters its queries to include only objects associated
  223. with the current :class:`~django.contrib.sites.models.Site`.
  224. Use :class:`~django.contrib.sites.managers.CurrentSiteManager` by adding it to
  225. your model explicitly. For example::
  226. from django.db import models
  227. from django.contrib.sites.models import Site
  228. from django.contrib.sites.managers import CurrentSiteManager
  229. class Photo(models.Model):
  230. photo = models.FileField(upload_to='/home/photos')
  231. photographer_name = models.CharField(max_length=100)
  232. pub_date = models.DateField()
  233. site = models.ForeignKey(Site)
  234. objects = models.Manager()
  235. on_site = CurrentSiteManager()
  236. With this model, ``Photo.objects.all()`` will return all ``Photo`` objects in
  237. the database, but ``Photo.on_site.all()`` will return only the ``Photo`` objects
  238. associated with the current site, according to the :setting:`SITE_ID` setting.
  239. Put another way, these two statements are equivalent::
  240. Photo.objects.filter(site=settings.SITE_ID)
  241. Photo.on_site.all()
  242. How did :class:`~django.contrib.sites.managers.CurrentSiteManager`
  243. know which field of ``Photo`` was the
  244. :class:`~django.contrib.sites.models.Site`? By default,
  245. :class:`~django.contrib.sites.managers.CurrentSiteManager` looks for a
  246. either a :class:`~django.db.models.ForeignKey` called
  247. ``site`` or a
  248. :class:`~django.db.models.ManyToManyField` called
  249. ``sites`` to filter on. If you use a field named something other than
  250. ``site`` or ``sites`` to identify which
  251. :class:`~django.contrib.sites.models.Site` objects your object is
  252. related to, then you need to explicitly pass the custom field name as
  253. a parameter to
  254. :class:`~django.contrib.sites.managers.CurrentSiteManager` on your
  255. model. The following model, which has a field called ``publish_on``,
  256. demonstrates this::
  257. from django.db import models
  258. from django.contrib.sites.models import Site
  259. from django.contrib.sites.managers import CurrentSiteManager
  260. class Photo(models.Model):
  261. photo = models.FileField(upload_to='/home/photos')
  262. photographer_name = models.CharField(max_length=100)
  263. pub_date = models.DateField()
  264. publish_on = models.ForeignKey(Site)
  265. objects = models.Manager()
  266. on_site = CurrentSiteManager('publish_on')
  267. If you attempt to use :class:`~django.contrib.sites.managers.CurrentSiteManager`
  268. and pass a field name that doesn't exist, Django will raise a ``ValueError``.
  269. Finally, note that you'll probably want to keep a normal
  270. (non-site-specific) ``Manager`` on your model, even if you use
  271. :class:`~django.contrib.sites.managers.CurrentSiteManager`. As
  272. explained in the :doc:`manager documentation </topics/db/managers>`, if
  273. you define a manager manually, then Django won't create the automatic
  274. ``objects = models.Manager()`` manager for you. Also note that certain
  275. parts of Django -- namely, the Django admin site and generic views --
  276. use whichever manager is defined *first* in the model, so if you want
  277. your admin site to have access to all objects (not just site-specific
  278. ones), put ``objects = models.Manager()`` in your model, before you
  279. define :class:`~django.contrib.sites.managers.CurrentSiteManager`.
  280. .. _site-middleware:
  281. Site middleware
  282. ===============
  283. .. versionadded:: 1.7
  284. If you often use this pattern::
  285. from django.contrib.sites.models import Site
  286. def my_view(request):
  287. site = Site.objects.get_current()
  288. ...
  289. there is simple way to avoid repetitions. Add
  290. :class:`django.contrib.sites.middleware.CurrentSiteMiddleware` to
  291. :setting:`MIDDLEWARE_CLASSES`. The middleware sets the ``site`` attribute on
  292. every request object, so you can use ``request.site`` to get the current site.
  293. How Django uses the sites framework
  294. ===================================
  295. Although it's not required that you use the sites framework, it's strongly
  296. encouraged, because Django takes advantage of it in a few places. Even if your
  297. Django installation is powering only a single site, you should take the two
  298. seconds to create the site object with your ``domain`` and ``name``, and point
  299. to its ID in your :setting:`SITE_ID` setting.
  300. Here's how Django uses the sites framework:
  301. * In the :mod:`redirects framework <django.contrib.redirects>`, each
  302. redirect object is associated with a particular site. When Django searches
  303. for a redirect, it takes into account the current site.
  304. * In the comments framework, each comment is associated with a particular
  305. site. When a comment is posted, its
  306. :class:`~django.contrib.sites.models.Site` is set to the current site,
  307. and when comments are listed via the appropriate template tag, only the
  308. comments for the current site are displayed.
  309. * In the :mod:`flatpages framework <django.contrib.flatpages>`, each
  310. flatpage is associated with a particular site. When a flatpage is created,
  311. you specify its :class:`~django.contrib.sites.models.Site`, and the
  312. :class:`~django.contrib.flatpages.middleware.FlatpageFallbackMiddleware`
  313. checks the current site in retrieving flatpages to display.
  314. * In the :mod:`syndication framework <django.contrib.syndication>`, the
  315. templates for ``title`` and ``description`` automatically have access to a
  316. variable ``{{ site }}``, which is the
  317. :class:`~django.contrib.sites.models.Site` object representing the current
  318. site. Also, the hook for providing item URLs will use the ``domain`` from
  319. the current :class:`~django.contrib.sites.models.Site` object if you don't
  320. specify a fully-qualified domain.
  321. * In the :mod:`authentication framework <django.contrib.auth>`, the
  322. :func:`django.contrib.auth.views.login` view passes the current
  323. :class:`~django.contrib.sites.models.Site` name to the template as
  324. ``{{ site_name }}``.
  325. * The shortcut view (``django.contrib.contenttypes.views.shortcut``)
  326. uses the domain of the current
  327. :class:`~django.contrib.sites.models.Site` object when calculating
  328. an object's URL.
  329. * In the admin framework, the "view on site" link uses the current
  330. :class:`~django.contrib.sites.models.Site` to work out the domain for the
  331. site that it will redirect to.
  332. ``RequestSite`` objects
  333. =======================
  334. .. _requestsite-objects:
  335. Some :doc:`django.contrib </ref/contrib/index>` applications take advantage of
  336. the sites framework but are architected in a way that doesn't *require* the
  337. sites framework to be installed in your database. (Some people don't want to,
  338. or just aren't *able* to install the extra database table that the sites
  339. framework requires.) For those cases, the framework provides a
  340. :class:`django.contrib.sites.requests.RequestSite` class, which can be used as
  341. a fallback when the database-backed sites framework is not available.
  342. .. class:: requests.RequestSite
  343. A class that shares the primary interface of
  344. :class:`~django.contrib.sites.models.Site` (i.e., it has
  345. ``domain`` and ``name`` attributes) but gets its data from a Django
  346. :class:`~django.http.HttpRequest` object rather than from a database.
  347. .. method:: __init__(request)
  348. Sets the ``name`` and ``domain`` attributes to the value of
  349. :meth:`~django.http.HttpRequest.get_host`.
  350. .. versionchanged:: 1.7
  351. This class used to be defined in ``django.contrib.sites.models``.
  352. A :class:`~django.contrib.sites.requests.RequestSite` object has a similar
  353. interface to a normal :class:`~django.contrib.sites.models.Site` object,
  354. except its :meth:`~django.contrib.sites.requests.RequestSite.__init__()`
  355. method takes an :class:`~django.http.HttpRequest` object. It's able to deduce
  356. the ``domain`` and ``name`` by looking at the request's domain. It has
  357. ``save()`` and ``delete()`` methods to match the interface of
  358. :class:`~django.contrib.sites.models.Site`, but the methods raise
  359. :exc:`~exceptions.NotImplementedError`..
  360. ``get_current_site`` shortcut
  361. =============================
  362. Finally, to avoid repetitive fallback code, the framework provides a
  363. :func:`django.contrib.sites.shortcuts.get_current_site` function.
  364. .. function:: shortcuts.get_current_site(request)
  365. A function that checks if ``django.contrib.sites`` is installed and
  366. returns either the current :class:`~django.contrib.sites.models.Site`
  367. object or a :class:`~django.contrib.sites.requests.RequestSite` object
  368. based on the request.
  369. .. versionchanged:: 1.7
  370. This function used to be defined in ``django.contrib.sites.models``.