sites.txt 20 KB

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