mirror of
https://github.com/django/django.git
synced 2025-01-10 10:26:34 +00:00
1168 lines
41 KiB
Plaintext
1168 lines
41 KiB
Plaintext
====================================================
|
||
The Django template language: for Python programmers
|
||
====================================================
|
||
|
||
.. currentmodule:: django.template
|
||
|
||
This document explains the Django template system from a technical
|
||
perspective -- how it works and how to extend it. If you're looking for
|
||
reference on the language syntax, see :doc:`/ref/templates/language`.
|
||
|
||
It assumes an understanding of templates, contexts, variables, tags, and
|
||
rendering. Start with the :ref:`introduction to the Django template language
|
||
<template-language-intro>` if you aren't familiar with these concepts.
|
||
|
||
Overview
|
||
========
|
||
|
||
Using the template system in Python is a three-step process:
|
||
|
||
1. You configure an :class:`Engine`.
|
||
2. You compile template code into a :class:`Template`.
|
||
3. You render the template with a :class:`Context`.
|
||
|
||
Django projects generally rely on the :ref:`high level, backend agnostic APIs
|
||
<template-engines>` for each of these steps instead of the template system's
|
||
lower level APIs:
|
||
|
||
1. For each :class:`~django.template.backends.django.DjangoTemplates` backend
|
||
in the :setting:`TEMPLATES` setting, Django instantiates an
|
||
:class:`Engine`. :class:`~django.template.backends.django.DjangoTemplates`
|
||
wraps :class:`Engine` and adapts it to the common template backend API.
|
||
2. The :mod:`django.template.loader` module provides functions such as
|
||
:func:`~django.template.loader.get_template` for loading templates. They
|
||
return a ``django.template.backends.django.Template`` which wraps the
|
||
actual :class:`django.template.Template`.
|
||
3. The ``Template`` obtained in the previous step has a
|
||
:meth:`~django.template.backends.base.Template.render` method which
|
||
marshals a context and possibly a request into a :class:`Context` and
|
||
delegates the rendering to the underlying :class:`Template`.
|
||
|
||
Configuring an engine
|
||
=====================
|
||
|
||
If you are using the :class:`~django.template.backends.django.DjangoTemplates`
|
||
backend, this probably isn't the documentation you're looking for. An instance
|
||
of the ``Engine`` class described below is accessible using the ``engine``
|
||
attribute of that backend and any attribute defaults mentioned below are
|
||
overridden by what's passed by
|
||
:class:`~django.template.backends.django.DjangoTemplates`.
|
||
|
||
.. class:: Engine(dirs=None, app_dirs=False, context_processors=None, debug=False, loaders=None, string_if_invalid='', file_charset='utf-8', libraries=None, builtins=None, autoescape=True)
|
||
|
||
When instantiating an ``Engine`` all arguments must be passed as keyword
|
||
arguments:
|
||
|
||
* ``dirs`` is a list of directories where the engine should look for
|
||
template source files. It is used to configure
|
||
:class:`filesystem.Loader <django.template.loaders.filesystem.Loader>`.
|
||
|
||
It defaults to an empty list.
|
||
|
||
* ``app_dirs`` only affects the default value of ``loaders``. See below.
|
||
|
||
It defaults to ``False``.
|
||
|
||
* ``autoescape`` controls whether HTML autoescaping is enabled.
|
||
|
||
It defaults to ``True``.
|
||
|
||
.. warning::
|
||
|
||
Only set it to ``False`` if you're rendering non-HTML templates!
|
||
|
||
* ``context_processors`` is a list of dotted Python paths to callables
|
||
that are used to populate the context when a template is rendered with a
|
||
request. These callables take a request object as their argument and
|
||
return a :class:`dict` of items to be merged into the context.
|
||
|
||
It defaults to an empty list.
|
||
|
||
See :class:`~django.template.RequestContext` for more information.
|
||
|
||
* ``debug`` is a boolean that turns on/off template debug mode. If it is
|
||
``True``, the template engine will store additional debug information
|
||
which can be used to display a detailed report for any exception raised
|
||
during template rendering.
|
||
|
||
It defaults to ``False``.
|
||
|
||
* ``loaders`` is a list of template loader classes, specified as strings.
|
||
Each ``Loader`` class knows how to import templates from a particular
|
||
source. Optionally, a tuple can be used instead of a string. The first
|
||
item in the tuple should be the ``Loader`` class name, subsequent items
|
||
are passed to the ``Loader`` during initialization.
|
||
|
||
It defaults to a list containing:
|
||
|
||
* ``'django.template.loaders.filesystem.Loader'``
|
||
* ``'django.template.loaders.app_directories.Loader'`` if and only if
|
||
``app_dirs`` is ``True``.
|
||
|
||
These loaders are then wrapped in
|
||
:class:`django.template.loaders.cached.Loader`.
|
||
|
||
See :ref:`template-loaders` for details.
|
||
|
||
* ``string_if_invalid`` is the output, as a string, that the template
|
||
system should use for invalid (e.g. misspelled) variables.
|
||
|
||
It defaults to the empty string.
|
||
|
||
See :ref:`invalid-template-variables` for details.
|
||
|
||
* ``file_charset`` is the charset used to read template files on disk.
|
||
|
||
It defaults to ``'utf-8'``.
|
||
|
||
* ``'libraries'``: A dictionary of labels and dotted Python paths of template
|
||
tag modules to register with the template engine. This is used to add new
|
||
libraries or provide alternate labels for existing ones. For example::
|
||
|
||
Engine(
|
||
libraries={
|
||
"myapp_tags": "path.to.myapp.tags",
|
||
"admin.urls": "django.contrib.admin.templatetags.admin_urls",
|
||
},
|
||
)
|
||
|
||
Libraries can be loaded by passing the corresponding dictionary key to
|
||
the :ttag:`{% load %}<load>` tag.
|
||
|
||
* ``'builtins'``: A list of dotted Python paths of template tag modules to
|
||
add to :doc:`built-ins </ref/templates/builtins>`. For example::
|
||
|
||
Engine(
|
||
builtins=["myapp.builtins"],
|
||
)
|
||
|
||
Tags and filters from built-in libraries can be used without first calling
|
||
the :ttag:`{% load %}<load>` tag.
|
||
|
||
.. staticmethod:: Engine.get_default()
|
||
|
||
Returns the underlying :class:`Engine` from the first configured
|
||
:class:`~django.template.backends.django.DjangoTemplates` engine. Raises
|
||
:exc:`~django.core.exceptions.ImproperlyConfigured` if no engines are
|
||
configured.
|
||
|
||
It's required for preserving APIs that rely on a globally available,
|
||
implicitly configured engine. Any other use is strongly discouraged.
|
||
|
||
.. method:: Engine.from_string(template_code)
|
||
|
||
Compiles the given template code and returns a :class:`Template` object.
|
||
|
||
.. method:: Engine.get_template(template_name)
|
||
|
||
Loads a template with the given name, compiles it and returns a
|
||
:class:`Template` object.
|
||
|
||
.. method:: Engine.select_template(template_name_list)
|
||
|
||
Like :meth:`~Engine.get_template`, except it takes a list of names
|
||
and returns the first template that was found.
|
||
|
||
Loading a template
|
||
==================
|
||
|
||
The recommended way to create a :class:`Template` is by calling the factory
|
||
methods of the :class:`Engine`: :meth:`~Engine.get_template`,
|
||
:meth:`~Engine.select_template` and :meth:`~Engine.from_string`.
|
||
|
||
In a Django project where the :setting:`TEMPLATES` setting defines a
|
||
:class:`~django.template.backends.django.DjangoTemplates` engine, it's
|
||
possible to instantiate a :class:`Template` directly. If more than one
|
||
:class:`~django.template.backends.django.DjangoTemplates` engine is defined,
|
||
the first one will be used.
|
||
|
||
.. class:: Template
|
||
|
||
This class lives at ``django.template.Template``. The constructor takes
|
||
one argument — the raw template code::
|
||
|
||
from django.template import Template
|
||
|
||
template = Template("My name is {{ my_name }}.")
|
||
|
||
.. admonition:: Behind the scenes
|
||
|
||
The system only parses your raw template code once -- when you create the
|
||
``Template`` object. From then on, it's stored internally as a tree
|
||
structure for performance.
|
||
|
||
Even the parsing itself is quite fast. Most of the parsing happens via a
|
||
single call to a single, short, regular expression.
|
||
|
||
Rendering a context
|
||
===================
|
||
|
||
Once you have a compiled :class:`Template` object, you can render a context
|
||
with it. You can reuse the same template to render it several times with
|
||
different contexts.
|
||
|
||
.. class:: Context(dict_=None, autoescape=True, use_l10n=None, use_tz=None)
|
||
|
||
The constructor of ``django.template.Context`` takes an optional argument —
|
||
a dictionary mapping variable names to variable values.
|
||
|
||
Three optional keyword arguments can also be specified:
|
||
|
||
* ``autoescape`` controls whether HTML autoescaping is enabled.
|
||
|
||
It defaults to ``True``.
|
||
|
||
.. warning::
|
||
|
||
Only set it to ``False`` if you're rendering non-HTML templates!
|
||
|
||
* ``use_l10n`` overrides whether values will be localized by default. If
|
||
set to ``True`` numbers and dates will be formatted based on locale.
|
||
|
||
It defaults to ``None``.
|
||
|
||
See :ref:`topic-l10n-templates` for details.
|
||
|
||
* ``use_tz`` overrides whether dates are converted to the local time when
|
||
rendered in a template. If set to ``True`` all dates will be rendered
|
||
using the local timezone. This takes precedence over :setting:`USE_TZ`.
|
||
|
||
It defaults to ``None``.
|
||
|
||
See :ref:`time-zones-in-templates` for details.
|
||
|
||
For example usage, see :ref:`playing-with-context` below.
|
||
|
||
.. method:: Template.render(context)
|
||
|
||
Call the :class:`Template` object's ``render()`` method with a
|
||
:class:`Context` to "fill" the template:
|
||
|
||
.. code-block:: pycon
|
||
|
||
>>> from django.template import Context, Template
|
||
>>> template = Template("My name is {{ my_name }}.")
|
||
|
||
>>> context = Context({"my_name": "Adrian"})
|
||
>>> template.render(context)
|
||
"My name is Adrian."
|
||
|
||
>>> context = Context({"my_name": "Dolores"})
|
||
>>> template.render(context)
|
||
"My name is Dolores."
|
||
|
||
Variables and lookups
|
||
---------------------
|
||
|
||
Variable names must consist of any letter (A-Z), any digit (0-9), an underscore
|
||
(but they must not start with an underscore) or a dot.
|
||
|
||
Dots have a special meaning in template rendering. A dot in a variable name
|
||
signifies a **lookup**. Specifically, when the template system encounters a
|
||
dot in a variable name, it tries the following lookups, in this order:
|
||
|
||
* Dictionary lookup. Example: ``foo["bar"]``
|
||
* Attribute lookup. Example: ``foo.bar``
|
||
* List-index lookup. Example: ``foo[bar]``
|
||
|
||
Note that "bar" in a template expression like ``{{ foo.bar }}`` will be
|
||
interpreted as a literal string and not using the value of the variable "bar",
|
||
if one exists in the template context.
|
||
|
||
The template system uses the first lookup type that works. It's short-circuit
|
||
logic. Here are a few examples:
|
||
|
||
.. code-block:: pycon
|
||
|
||
>>> from django.template import Context, Template
|
||
>>> t = Template("My name is {{ person.first_name }}.")
|
||
>>> d = {"person": {"first_name": "Joe", "last_name": "Johnson"}}
|
||
>>> t.render(Context(d))
|
||
"My name is Joe."
|
||
|
||
>>> class PersonClass:
|
||
... pass
|
||
...
|
||
>>> p = PersonClass()
|
||
>>> p.first_name = "Ron"
|
||
>>> p.last_name = "Nasty"
|
||
>>> t.render(Context({"person": p}))
|
||
"My name is Ron."
|
||
|
||
>>> t = Template("The first stooge in the list is {{ stooges.0 }}.")
|
||
>>> c = Context({"stooges": ["Larry", "Curly", "Moe"]})
|
||
>>> t.render(c)
|
||
"The first stooge in the list is Larry."
|
||
|
||
If any part of the variable is callable, the template system will try calling
|
||
it. Example:
|
||
|
||
.. code-block:: pycon
|
||
|
||
>>> class PersonClass2:
|
||
... def name(self):
|
||
... return "Samantha"
|
||
...
|
||
>>> t = Template("My name is {{ person.name }}.")
|
||
>>> t.render(Context({"person": PersonClass2}))
|
||
"My name is Samantha."
|
||
|
||
Callable variables are slightly more complex than variables which only require
|
||
straight lookups. Here are some things to keep in mind:
|
||
|
||
* If the variable raises an exception when called, the exception will be
|
||
propagated, unless the exception has an attribute
|
||
``silent_variable_failure`` whose value is ``True``. If the exception
|
||
*does* have a ``silent_variable_failure`` attribute whose value is
|
||
``True``, the variable will render as the value of the engine's
|
||
``string_if_invalid`` configuration option (an empty string, by default).
|
||
Example:
|
||
|
||
.. code-block:: pycon
|
||
|
||
>>> t = Template("My name is {{ person.first_name }}.")
|
||
>>> class PersonClass3:
|
||
... def first_name(self):
|
||
... raise AssertionError("foo")
|
||
...
|
||
>>> p = PersonClass3()
|
||
>>> t.render(Context({"person": p}))
|
||
Traceback (most recent call last):
|
||
...
|
||
AssertionError: foo
|
||
|
||
>>> class SilentAssertionError(Exception):
|
||
... silent_variable_failure = True
|
||
...
|
||
>>> class PersonClass4:
|
||
... def first_name(self):
|
||
... raise SilentAssertionError
|
||
...
|
||
>>> p = PersonClass4()
|
||
>>> t.render(Context({"person": p}))
|
||
"My name is ."
|
||
|
||
Note that :exc:`django.core.exceptions.ObjectDoesNotExist`, which is the
|
||
base class for all Django database API ``DoesNotExist`` exceptions, has
|
||
``silent_variable_failure = True``. So if you're using Django templates
|
||
with Django model objects, any ``DoesNotExist`` exception will fail
|
||
silently.
|
||
|
||
* A variable can only be called if it has no required arguments. Otherwise,
|
||
the system will return the value of the engine's ``string_if_invalid``
|
||
option.
|
||
|
||
.. _alters-data-description:
|
||
|
||
* There can be side effects when calling some variables, and it'd be either
|
||
foolish or a security hole to allow the template system to access them.
|
||
|
||
A good example is the :meth:`~django.db.models.Model.delete` method on
|
||
each Django model object. The template system shouldn't be allowed to do
|
||
something like this:
|
||
|
||
.. code-block:: html+django
|
||
|
||
I will now delete this valuable data. {{ data.delete }}
|
||
|
||
To prevent this, set an ``alters_data`` attribute on the callable
|
||
variable. The template system won't call a variable if it has
|
||
``alters_data=True`` set, and will instead replace the variable with
|
||
``string_if_invalid``, unconditionally. The
|
||
dynamically-generated :meth:`~django.db.models.Model.delete` and
|
||
:meth:`~django.db.models.Model.save` methods on Django model objects get
|
||
``alters_data=True`` automatically. Example::
|
||
|
||
def sensitive_function(self):
|
||
self.database_record.delete()
|
||
|
||
|
||
sensitive_function.alters_data = True
|
||
|
||
* Occasionally you may want to turn off this feature for other reasons,
|
||
and tell the template system to leave a variable uncalled no matter
|
||
what. To do so, set a ``do_not_call_in_templates`` attribute on the
|
||
callable with the value ``True``. The template system then will act as
|
||
if your variable is not callable (allowing you to access attributes of
|
||
the callable, for example).
|
||
|
||
.. _invalid-template-variables:
|
||
|
||
How invalid variables are handled
|
||
---------------------------------
|
||
|
||
Generally, if a variable doesn't exist, the template system inserts the value
|
||
of the engine's ``string_if_invalid`` configuration option, which is set to
|
||
``''`` (the empty string) by default.
|
||
|
||
Filters that are applied to an invalid variable will only be applied if
|
||
``string_if_invalid`` is set to ``''`` (the empty string). If
|
||
``string_if_invalid`` is set to any other value, variable filters will be
|
||
ignored.
|
||
|
||
This behavior is slightly different for the ``if``, ``for`` and ``regroup``
|
||
template tags. If an invalid variable is provided to one of these template
|
||
tags, the variable will be interpreted as ``None``. Filters are always
|
||
applied to invalid variables within these template tags.
|
||
|
||
If ``string_if_invalid`` contains a ``'%s'``, the format marker will be
|
||
replaced with the name of the invalid variable.
|
||
|
||
.. admonition:: For debug purposes only!
|
||
|
||
While ``string_if_invalid`` can be a useful debugging tool, it is a bad
|
||
idea to turn it on as a 'development default'.
|
||
|
||
Many templates, including some of Django's, rely upon the silence of the
|
||
template system when a nonexistent variable is encountered. If you assign a
|
||
value other than ``''`` to ``string_if_invalid``, you will experience
|
||
rendering problems with these templates and sites.
|
||
|
||
Generally, ``string_if_invalid`` should only be enabled in order to debug
|
||
a specific template problem, then cleared once debugging is complete.
|
||
|
||
Built-in variables
|
||
------------------
|
||
|
||
Every context contains ``True``, ``False`` and ``None``. As you would expect,
|
||
these variables resolve to the corresponding Python objects.
|
||
|
||
Limitations with string literals
|
||
--------------------------------
|
||
|
||
Django's template language has no way to escape the characters used for its own
|
||
syntax. For example, the :ttag:`templatetag` tag is required if you need to
|
||
output character sequences like ``{%`` and ``%}``.
|
||
|
||
A similar issue exists if you want to include these sequences in template filter
|
||
or tag arguments. For example, when parsing a block tag, Django's template
|
||
parser looks for the first occurrence of ``%}`` after a ``{%``. This prevents
|
||
the use of ``"%}"`` as a string literal. For example, a ``TemplateSyntaxError``
|
||
will be raised for the following expressions:
|
||
|
||
.. code-block:: html+django
|
||
|
||
{% include "template.html" tvar="Some string literal with %} in it." %}
|
||
|
||
{% with tvar="Some string literal with %} in it." %}{% endwith %}
|
||
|
||
The same issue can be triggered by using a reserved sequence in filter
|
||
arguments:
|
||
|
||
.. code-block:: html+django
|
||
|
||
{{ some.variable|default:"}}" }}
|
||
|
||
If you need to use strings with these sequences, store them in template
|
||
variables or use a custom template tag or filter to workaround the limitation.
|
||
|
||
.. _playing-with-context:
|
||
|
||
Playing with ``Context`` objects
|
||
================================
|
||
|
||
Most of the time, you'll instantiate :class:`Context` objects by passing in a
|
||
fully-populated dictionary to ``Context()``. But you can add and delete items
|
||
from a ``Context`` object once it's been instantiated, too, using standard
|
||
dictionary syntax:
|
||
|
||
.. code-block:: pycon
|
||
|
||
>>> from django.template import Context
|
||
>>> c = Context({"foo": "bar"})
|
||
>>> c["foo"]
|
||
'bar'
|
||
>>> del c["foo"]
|
||
>>> c["foo"]
|
||
Traceback (most recent call last):
|
||
...
|
||
KeyError: 'foo'
|
||
>>> c["newvariable"] = "hello"
|
||
>>> c["newvariable"]
|
||
'hello'
|
||
|
||
.. method:: Context.get(key, otherwise=None)
|
||
|
||
Returns the value for ``key`` if ``key`` is in the context, else returns
|
||
``otherwise``.
|
||
|
||
.. method:: Context.setdefault(key, default=None)
|
||
|
||
If ``key`` is in the context, returns its value. Otherwise inserts ``key``
|
||
with a value of ``default`` and returns ``default``.
|
||
|
||
.. method:: Context.pop()
|
||
.. method:: Context.push()
|
||
.. exception:: ContextPopException
|
||
|
||
A ``Context`` object is a stack. That is, you can ``push()`` and ``pop()`` it.
|
||
If you ``pop()`` too much, it'll raise
|
||
``django.template.ContextPopException``:
|
||
|
||
.. code-block:: pycon
|
||
|
||
>>> c = Context()
|
||
>>> c["foo"] = "first level"
|
||
>>> c.push()
|
||
{}
|
||
>>> c["foo"] = "second level"
|
||
>>> c["foo"]
|
||
'second level'
|
||
>>> c.pop()
|
||
{'foo': 'second level'}
|
||
>>> c["foo"]
|
||
'first level'
|
||
>>> c["foo"] = "overwritten"
|
||
>>> c["foo"]
|
||
'overwritten'
|
||
>>> c.pop()
|
||
Traceback (most recent call last):
|
||
...
|
||
ContextPopException
|
||
|
||
You can also use ``push()`` as a context manager to ensure a matching ``pop()``
|
||
is called.
|
||
|
||
.. code-block:: pycon
|
||
|
||
>>> c = Context()
|
||
>>> c["foo"] = "first level"
|
||
>>> with c.push():
|
||
... c["foo"] = "second level"
|
||
... c["foo"]
|
||
...
|
||
'second level'
|
||
>>> c["foo"]
|
||
'first level'
|
||
|
||
All arguments passed to ``push()`` will be passed to the ``dict`` constructor
|
||
used to build the new context level.
|
||
|
||
.. code-block:: pycon
|
||
|
||
>>> c = Context()
|
||
>>> c["foo"] = "first level"
|
||
>>> with c.push(foo="second level"):
|
||
... c["foo"]
|
||
...
|
||
'second level'
|
||
>>> c["foo"]
|
||
'first level'
|
||
|
||
.. method:: Context.update(other_dict)
|
||
|
||
In addition to ``push()`` and ``pop()``, the ``Context``
|
||
object also defines an ``update()`` method. This works like ``push()``
|
||
but takes a dictionary as an argument and pushes that dictionary onto
|
||
the stack instead of an empty one.
|
||
|
||
.. code-block:: pycon
|
||
|
||
>>> c = Context()
|
||
>>> c["foo"] = "first level"
|
||
>>> c.update({"foo": "updated"})
|
||
{'foo': 'updated'}
|
||
>>> c["foo"]
|
||
'updated'
|
||
>>> c.pop()
|
||
{'foo': 'updated'}
|
||
>>> c["foo"]
|
||
'first level'
|
||
|
||
Like ``push()``, you can use ``update()`` as a context manager to ensure a
|
||
matching ``pop()`` is called.
|
||
|
||
.. code-block:: pycon
|
||
|
||
>>> c = Context()
|
||
>>> c["foo"] = "first level"
|
||
>>> with c.update({"foo": "second level"}):
|
||
... c["foo"]
|
||
...
|
||
'second level'
|
||
>>> c["foo"]
|
||
'first level'
|
||
|
||
Using a ``Context`` as a stack comes in handy in :ref:`some custom template
|
||
tags <howto-writing-custom-template-tags>`.
|
||
|
||
.. method:: Context.flatten()
|
||
|
||
Using ``flatten()`` method you can get whole ``Context`` stack as one dictionary
|
||
including builtin variables.
|
||
|
||
.. code-block:: pycon
|
||
|
||
>>> c = Context()
|
||
>>> c["foo"] = "first level"
|
||
>>> c.update({"bar": "second level"})
|
||
{'bar': 'second level'}
|
||
>>> c.flatten()
|
||
{'True': True, 'None': None, 'foo': 'first level', 'False': False, 'bar': 'second level'}
|
||
|
||
A ``flatten()`` method is also internally used to make ``Context`` objects comparable.
|
||
|
||
.. code-block:: pycon
|
||
|
||
>>> c1 = Context()
|
||
>>> c1["foo"] = "first level"
|
||
>>> c1["bar"] = "second level"
|
||
>>> c2 = Context()
|
||
>>> c2.update({"bar": "second level", "foo": "first level"})
|
||
{'foo': 'first level', 'bar': 'second level'}
|
||
>>> c1 == c2
|
||
True
|
||
|
||
Result from ``flatten()`` can be useful in unit tests to compare ``Context``
|
||
against ``dict``::
|
||
|
||
class ContextTest(unittest.TestCase):
|
||
def test_against_dictionary(self):
|
||
c1 = Context()
|
||
c1["update"] = "value"
|
||
self.assertEqual(
|
||
c1.flatten(),
|
||
{
|
||
"True": True,
|
||
"None": None,
|
||
"False": False,
|
||
"update": "value",
|
||
},
|
||
)
|
||
|
||
.. _subclassing-context-requestcontext:
|
||
|
||
Using ``RequestContext``
|
||
------------------------
|
||
|
||
.. class:: RequestContext(request, dict_=None, processors=None, use_l10n=None, use_tz=None, autoescape=True)
|
||
|
||
Django comes with a special :class:`~django.template.Context` class,
|
||
``django.template.RequestContext``, that acts slightly differently from the
|
||
normal ``django.template.Context``. The first difference is that it takes an
|
||
:class:`~django.http.HttpRequest` as its first argument. For example::
|
||
|
||
c = RequestContext(
|
||
request,
|
||
{
|
||
"foo": "bar",
|
||
},
|
||
)
|
||
|
||
The second difference is that it automatically populates the context with a
|
||
few variables, according to the engine's ``context_processors`` configuration
|
||
option.
|
||
|
||
The ``context_processors`` option is a list of callables -- called **context
|
||
processors** -- that take a request object as their argument and return a
|
||
dictionary of items to be merged into the context. In the default generated
|
||
settings file, the default template engine contains the following context
|
||
processors::
|
||
|
||
[
|
||
"django.template.context_processors.request",
|
||
"django.contrib.auth.context_processors.auth",
|
||
"django.contrib.messages.context_processors.messages",
|
||
]
|
||
|
||
In addition to these, :class:`RequestContext` always enables
|
||
``'django.template.context_processors.csrf'``. This is a security related
|
||
context processor required by the admin and other contrib apps, and, in case
|
||
of accidental misconfiguration, it is deliberately hardcoded in and cannot be
|
||
turned off in the ``context_processors`` option.
|
||
|
||
Each processor is applied in order. That means, if one processor adds a
|
||
variable to the context and a second processor adds a variable with the same
|
||
name, the second will override the first. The default processors are explained
|
||
below.
|
||
|
||
.. admonition:: When context processors are applied
|
||
|
||
Context processors are applied on top of context data. This means that a
|
||
context processor may overwrite variables you've supplied to your
|
||
:class:`Context` or :class:`RequestContext`, so take care to avoid
|
||
variable names that overlap with those supplied by your context
|
||
processors.
|
||
|
||
If you want context data to take priority over context processors, use the
|
||
following pattern::
|
||
|
||
from django.template import RequestContext
|
||
|
||
request_context = RequestContext(request)
|
||
request_context.push({"my_name": "Adrian"})
|
||
|
||
Django does this to allow context data to override context processors in
|
||
APIs such as :func:`~django.shortcuts.render` and
|
||
:class:`~django.template.response.TemplateResponse`.
|
||
|
||
Also, you can give :class:`RequestContext` a list of additional processors,
|
||
using the optional, third positional argument, ``processors``. In this
|
||
example, the :class:`RequestContext` instance gets an ``ip_address`` variable::
|
||
|
||
from django.http import HttpResponse
|
||
from django.template import RequestContext, Template
|
||
|
||
|
||
def ip_address_processor(request):
|
||
return {"ip_address": request.META["REMOTE_ADDR"]}
|
||
|
||
|
||
def client_ip_view(request):
|
||
template = Template("{{ title }}: {{ ip_address }}")
|
||
context = RequestContext(
|
||
request,
|
||
{
|
||
"title": "Your IP Address",
|
||
},
|
||
[ip_address_processor],
|
||
)
|
||
return HttpResponse(template.render(context))
|
||
|
||
.. _context-processors:
|
||
|
||
Built-in template context processors
|
||
------------------------------------
|
||
|
||
Here's what each of the built-in processors does:
|
||
|
||
.. currentmodule:: django.contrib.auth.context_processors
|
||
|
||
``django.contrib.auth.context_processors.auth``
|
||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
||
.. function:: auth(request)
|
||
|
||
If this processor is enabled, every ``RequestContext`` will contain these
|
||
variables:
|
||
|
||
* ``user`` -- An ``auth.User`` instance representing the currently
|
||
logged-in user (or an ``AnonymousUser`` instance, if the client isn't
|
||
logged in).
|
||
|
||
* ``perms`` -- An instance of
|
||
``django.contrib.auth.context_processors.PermWrapper``, representing the
|
||
permissions that the currently logged-in user has.
|
||
|
||
.. currentmodule:: django.template.context_processors
|
||
|
||
``django.template.context_processors.debug``
|
||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
||
.. function:: debug(request)
|
||
|
||
If this processor is enabled, every ``RequestContext`` will contain these two
|
||
variables -- but only if your :setting:`DEBUG` setting is set to ``True`` and
|
||
the request's IP address (``request.META['REMOTE_ADDR']``) is in the
|
||
:setting:`INTERNAL_IPS` setting:
|
||
|
||
* ``debug`` -- ``True``. You can use this in templates to test whether
|
||
you're in :setting:`DEBUG` mode.
|
||
* ``sql_queries`` -- A list of ``{'sql': ..., 'time': ...}`` dictionaries,
|
||
representing every SQL query that has happened so far during the request
|
||
and how long it took. The list is in order by database alias and then by
|
||
query. It's lazily generated on access.
|
||
|
||
``django.template.context_processors.i18n``
|
||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
||
.. function:: i18n(request)
|
||
|
||
If this processor is enabled, every ``RequestContext`` will contain these
|
||
variables:
|
||
|
||
* ``LANGUAGES`` -- The value of the :setting:`LANGUAGES` setting.
|
||
* ``LANGUAGE_BIDI`` -- ``True`` if the current language is a right-to-left
|
||
language, e.g. Hebrew, Arabic. ``False`` if it's a left-to-right language,
|
||
e.g. English, French, German.
|
||
* ``LANGUAGE_CODE`` -- ``request.LANGUAGE_CODE``, if it exists. Otherwise,
|
||
the value of the :setting:`LANGUAGE_CODE` setting.
|
||
|
||
See :ref:`i18n template tags <i18n-template-tags>` for template tags that
|
||
generate the same values.
|
||
|
||
``django.template.context_processors.media``
|
||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
||
If this processor is enabled, every ``RequestContext`` will contain a variable
|
||
``MEDIA_URL``, providing the value of the :setting:`MEDIA_URL` setting.
|
||
|
||
``django.template.context_processors.static``
|
||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
||
.. function:: static(request)
|
||
|
||
If this processor is enabled, every ``RequestContext`` will contain a variable
|
||
``STATIC_URL``, providing the value of the :setting:`STATIC_URL` setting.
|
||
|
||
``django.template.context_processors.csrf``
|
||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
||
This processor adds a token that is needed by the :ttag:`csrf_token` template
|
||
tag for protection against :doc:`Cross Site Request Forgeries
|
||
</ref/csrf>`.
|
||
|
||
``django.template.context_processors.request``
|
||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
||
If this processor is enabled, every ``RequestContext`` will contain a variable
|
||
``request``, which is the current :class:`~django.http.HttpRequest`.
|
||
|
||
``django.template.context_processors.tz``
|
||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
||
.. function:: tz(request)
|
||
|
||
If this processor is enabled, every ``RequestContext`` will contain a variable
|
||
``TIME_ZONE``, providing the name of the currently active time zone.
|
||
|
||
``django.contrib.messages.context_processors.messages``
|
||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
||
If this processor is enabled, every ``RequestContext`` will contain these two
|
||
variables:
|
||
|
||
* ``messages`` -- A list of messages (as strings) that have been set
|
||
via the :doc:`messages framework </ref/contrib/messages>`.
|
||
* ``DEFAULT_MESSAGE_LEVELS`` -- A mapping of the message level names to
|
||
:ref:`their numeric value <message-level-constants>`.
|
||
|
||
Writing your own context processors
|
||
-----------------------------------
|
||
|
||
A context processor has a simple interface: It's a Python function that takes
|
||
one argument, an :class:`~django.http.HttpRequest` object, and returns a
|
||
dictionary that gets added to the template context.
|
||
|
||
For example, to add the :setting:`DEFAULT_FROM_EMAIL` setting to every
|
||
context::
|
||
|
||
from django.conf import settings
|
||
|
||
|
||
def from_email(request):
|
||
return {
|
||
"DEFAULT_FROM_EMAIL": settings.DEFAULT_FROM_EMAIL,
|
||
}
|
||
|
||
Custom context processors can live anywhere in your code base. All Django
|
||
cares about is that your custom context processors are pointed to by the
|
||
``'context_processors'`` option in your :setting:`TEMPLATES` setting — or the
|
||
``context_processors`` argument of :class:`~django.template.Engine` if you're
|
||
using it directly.
|
||
|
||
Loading templates
|
||
=================
|
||
|
||
Generally, you'll store templates in files on your filesystem rather than
|
||
using the low-level :class:`~django.template.Template` API yourself. Save
|
||
templates in a directory specified as a **template directory**.
|
||
|
||
Django searches for template directories in a number of places, depending on
|
||
your template loading settings (see "Loader types" below), but the most basic
|
||
way of specifying template directories is by using the :setting:`DIRS
|
||
<TEMPLATES-DIRS>` option.
|
||
|
||
The :setting:`DIRS <TEMPLATES-DIRS>` option
|
||
-------------------------------------------
|
||
|
||
Tell Django what your template directories are by using the :setting:`DIRS
|
||
<TEMPLATES-DIRS>` option in the :setting:`TEMPLATES` setting in your settings
|
||
file — or the ``dirs`` argument of :class:`~django.template.Engine`. This
|
||
should be set to a list of strings that contain full paths to your template
|
||
directories::
|
||
|
||
TEMPLATES = [
|
||
{
|
||
"BACKEND": "django.template.backends.django.DjangoTemplates",
|
||
"DIRS": [
|
||
"/home/html/templates/lawrence.com",
|
||
"/home/html/templates/default",
|
||
],
|
||
},
|
||
]
|
||
|
||
Your templates can go anywhere you want, as long as the directories and
|
||
templates are readable by the web server. They can have any extension you want,
|
||
such as ``.html`` or ``.txt``, or they can have no extension at all.
|
||
|
||
Note that these paths should use Unix-style forward slashes, even on Windows.
|
||
|
||
.. _template-loaders:
|
||
|
||
Loader types
|
||
------------
|
||
|
||
By default, Django uses a filesystem-based template loader, but Django comes
|
||
with a few other template loaders, which know how to load templates from other
|
||
sources.
|
||
|
||
Some of these other loaders are disabled by default, but you can activate them
|
||
by adding a ``'loaders'`` option to your ``DjangoTemplates`` backend in the
|
||
:setting:`TEMPLATES` setting or passing a ``loaders`` argument to
|
||
:class:`~django.template.Engine`. ``loaders`` should be a list of strings or
|
||
tuples, where each represents a template loader class. Here are the template
|
||
loaders that come with Django:
|
||
|
||
.. currentmodule:: django.template.loaders
|
||
|
||
``django.template.loaders.filesystem.Loader``
|
||
|
||
.. class:: filesystem.Loader
|
||
|
||
Loads templates from the filesystem, according to
|
||
:setting:`DIRS <TEMPLATES-DIRS>`.
|
||
|
||
This loader is enabled by default. However it won't find any templates
|
||
until you set :setting:`DIRS <TEMPLATES-DIRS>` to a non-empty list::
|
||
|
||
TEMPLATES = [
|
||
{
|
||
"BACKEND": "django.template.backends.django.DjangoTemplates",
|
||
"DIRS": [BASE_DIR / "templates"],
|
||
}
|
||
]
|
||
|
||
You can also override ``'DIRS'`` and specify specific directories for a
|
||
particular filesystem loader::
|
||
|
||
TEMPLATES = [
|
||
{
|
||
"BACKEND": "django.template.backends.django.DjangoTemplates",
|
||
"OPTIONS": {
|
||
"loaders": [
|
||
(
|
||
"django.template.loaders.filesystem.Loader",
|
||
[BASE_DIR / "templates"],
|
||
),
|
||
],
|
||
},
|
||
}
|
||
]
|
||
|
||
``django.template.loaders.app_directories.Loader``
|
||
|
||
.. class:: app_directories.Loader
|
||
|
||
Loads templates from Django apps on the filesystem. For each app in
|
||
:setting:`INSTALLED_APPS`, the loader looks for a ``templates``
|
||
subdirectory. If the directory exists, Django looks for templates in there.
|
||
|
||
This means you can store templates with your individual apps. This also
|
||
helps to distribute Django apps with default templates.
|
||
|
||
For example, for this setting::
|
||
|
||
INSTALLED_APPS = ["myproject.polls", "myproject.music"]
|
||
|
||
...then ``get_template('foo.html')`` will look for ``foo.html`` in these
|
||
directories, in this order:
|
||
|
||
* ``/path/to/myproject/polls/templates/``
|
||
* ``/path/to/myproject/music/templates/``
|
||
|
||
... and will use the one it finds first.
|
||
|
||
The order of :setting:`INSTALLED_APPS` is significant! For example, if you
|
||
want to customize the Django admin, you might choose to override the
|
||
standard ``admin/base_site.html`` template, from ``django.contrib.admin``,
|
||
with your own ``admin/base_site.html`` in ``myproject.polls``. You must
|
||
then make sure that your ``myproject.polls`` comes *before*
|
||
``django.contrib.admin`` in :setting:`INSTALLED_APPS`, otherwise
|
||
``django.contrib.admin``’s will be loaded first and yours will be ignored.
|
||
|
||
Note that the loader performs an optimization when it first runs:
|
||
it caches a list of which :setting:`INSTALLED_APPS` packages have a
|
||
``templates`` subdirectory.
|
||
|
||
You can enable this loader by setting :setting:`APP_DIRS
|
||
<TEMPLATES-APP_DIRS>` to ``True``::
|
||
|
||
TEMPLATES = [
|
||
{
|
||
"BACKEND": "django.template.backends.django.DjangoTemplates",
|
||
"APP_DIRS": True,
|
||
}
|
||
]
|
||
|
||
``django.template.loaders.cached.Loader``
|
||
|
||
.. class:: cached.Loader
|
||
|
||
While the Django template system is quite fast, if it needs to read and
|
||
compile your templates every time they're rendered, the overhead from that
|
||
can add up.
|
||
|
||
You configure the cached template loader with a list of other loaders that
|
||
it should wrap. The wrapped loaders are used to locate unknown templates
|
||
when they're first encountered. The cached loader then stores the compiled
|
||
``Template`` in memory. The cached ``Template`` instance is returned for
|
||
subsequent requests to load the same template.
|
||
|
||
This loader is automatically enabled if :setting:`OPTIONS['loaders']
|
||
<TEMPLATES-OPTIONS>` isn't specified.
|
||
|
||
You can manually specify template caching with some custom template loaders
|
||
using settings like this::
|
||
|
||
TEMPLATES = [
|
||
{
|
||
"BACKEND": "django.template.backends.django.DjangoTemplates",
|
||
"DIRS": [BASE_DIR / "templates"],
|
||
"OPTIONS": {
|
||
"loaders": [
|
||
(
|
||
"django.template.loaders.cached.Loader",
|
||
[
|
||
"django.template.loaders.filesystem.Loader",
|
||
"django.template.loaders.app_directories.Loader",
|
||
"path.to.custom.Loader",
|
||
],
|
||
),
|
||
],
|
||
},
|
||
}
|
||
]
|
||
|
||
.. note::
|
||
|
||
All of the built-in Django template tags are safe to use with the
|
||
cached loader, but if you're using custom template tags that come from
|
||
third party packages, or that you wrote yourself, you should ensure
|
||
that the ``Node`` implementation for each tag is thread-safe. For more
|
||
information, see :ref:`template tag thread safety considerations
|
||
<template_tag_thread_safety>`.
|
||
|
||
``django.template.loaders.locmem.Loader``
|
||
|
||
.. class:: locmem.Loader
|
||
|
||
Loads templates from a Python dictionary. This is useful for testing.
|
||
|
||
This loader takes a dictionary of templates as its first argument::
|
||
|
||
TEMPLATES = [
|
||
{
|
||
"BACKEND": "django.template.backends.django.DjangoTemplates",
|
||
"OPTIONS": {
|
||
"loaders": [
|
||
(
|
||
"django.template.loaders.locmem.Loader",
|
||
{
|
||
"index.html": "content here",
|
||
},
|
||
),
|
||
],
|
||
},
|
||
}
|
||
]
|
||
|
||
This loader is disabled by default.
|
||
|
||
Django uses the template loaders in order according to the ``'loaders'``
|
||
option. It uses each loader until a loader finds a match.
|
||
|
||
.. _custom-template-loaders:
|
||
|
||
.. currentmodule:: django.template.loaders.base
|
||
|
||
Custom loaders
|
||
==============
|
||
|
||
It's possible to load templates from additional sources using custom template
|
||
loaders. Custom ``Loader`` classes should inherit from
|
||
``django.template.loaders.base.Loader`` and define the ``get_contents()`` and
|
||
``get_template_sources()`` methods.
|
||
|
||
Loader methods
|
||
--------------
|
||
|
||
.. class:: Loader
|
||
|
||
Loads templates from a given source, such as the filesystem or a database.
|
||
|
||
.. method:: get_template_sources(template_name)
|
||
|
||
A method that takes a ``template_name`` and yields
|
||
:class:`~django.template.base.Origin` instances for each possible
|
||
source.
|
||
|
||
For example, the filesystem loader may receive ``'index.html'`` as a
|
||
``template_name`` argument. This method would yield origins for the
|
||
full path of ``index.html`` as it appears in each template directory
|
||
the loader looks at.
|
||
|
||
The method doesn't need to verify that the template exists at a given
|
||
path, but it should ensure the path is valid. For instance, the
|
||
filesystem loader makes sure the path lies under a valid template
|
||
directory.
|
||
|
||
.. method:: get_contents(origin)
|
||
|
||
Returns the contents for a template given a
|
||
:class:`~django.template.base.Origin` instance.
|
||
|
||
This is where a filesystem loader would read contents from the
|
||
filesystem, or a database loader would read from the database. If a
|
||
matching template doesn't exist, this should raise a
|
||
:exc:`~django.template.TemplateDoesNotExist` error.
|
||
|
||
.. method:: get_template(template_name, skip=None)
|
||
|
||
Returns a ``Template`` object for a given ``template_name`` by looping
|
||
through results from :meth:`get_template_sources` and calling
|
||
:meth:`get_contents`. This returns the first matching template. If no
|
||
template is found, :exc:`~django.template.TemplateDoesNotExist` is
|
||
raised.
|
||
|
||
The optional ``skip`` argument is a list of origins to ignore when
|
||
extending templates. This allow templates to extend other templates of
|
||
the same name. It also used to avoid recursion errors.
|
||
|
||
In general, it is enough to define :meth:`get_template_sources` and
|
||
:meth:`get_contents` for custom template loaders. ``get_template()``
|
||
will usually not need to be overridden.
|
||
|
||
.. admonition:: Building your own
|
||
|
||
For examples, read the :source:`source code for Django's built-in loaders
|
||
<django/template/loaders>`.
|
||
|
||
.. currentmodule:: django.template.base
|
||
|
||
Template origin
|
||
===============
|
||
|
||
Templates have an ``origin`` containing attributes depending on the source
|
||
they are loaded from.
|
||
|
||
.. class:: Origin(name, template_name=None, loader=None)
|
||
|
||
.. attribute:: name
|
||
|
||
The path to the template as returned by the template loader.
|
||
For loaders that read from the file system, this is the full
|
||
path to the template.
|
||
|
||
If the template is instantiated directly rather than through a
|
||
template loader, this is a string value of ``<unknown_source>``.
|
||
|
||
.. attribute:: template_name
|
||
|
||
The relative path to the template as passed into the
|
||
template loader.
|
||
|
||
If the template is instantiated directly rather than through a
|
||
template loader, this is ``None``.
|
||
|
||
.. attribute:: loader
|
||
|
||
The template loader instance that constructed this ``Origin``.
|
||
|
||
If the template is instantiated directly rather than through a
|
||
template loader, this is ``None``.
|
||
|
||
:class:`django.template.loaders.cached.Loader` requires all of its
|
||
wrapped loaders to set this attribute, typically by instantiating
|
||
the ``Origin`` with ``loader=self``.
|