2017-01-19 04:30:21 +08:00
|
|
|
import functools
|
|
|
|
|
2014-11-14 22:48:27 +08:00
|
|
|
from django.core.exceptions import ImproperlyConfigured
|
|
|
|
from django.utils.functional import cached_property
|
2014-11-20 06:23:58 +08:00
|
|
|
from django.utils.module_loading import import_string
|
2014-11-14 22:48:27 +08:00
|
|
|
|
2019-07-18 14:13:01 +08:00
|
|
|
from .base import Template
|
|
|
|
from .context import Context, _builtin_context_processors
|
2015-04-25 03:35:30 +08:00
|
|
|
from .exceptions import TemplateDoesNotExist
|
2015-05-09 04:10:36 +08:00
|
|
|
from .library import import_library
|
2014-11-14 22:48:27 +08:00
|
|
|
|
|
|
|
|
2017-01-19 15:39:46 +08:00
|
|
|
class Engine:
|
2015-05-09 04:10:36 +08:00
|
|
|
default_builtins = [
|
|
|
|
'django.template.defaulttags',
|
|
|
|
'django.template.defaultfilters',
|
|
|
|
'django.template.loader_tags',
|
|
|
|
]
|
2014-11-14 22:48:27 +08:00
|
|
|
|
2015-08-17 21:34:50 +08:00
|
|
|
def __init__(self, dirs=None, app_dirs=False, context_processors=None,
|
2014-11-23 04:33:40 +08:00
|
|
|
debug=False, loaders=None, string_if_invalid='',
|
2015-11-08 17:06:07 +08:00
|
|
|
file_charset='utf-8', libraries=None, builtins=None, autoescape=True):
|
2014-11-14 22:48:27 +08:00
|
|
|
if dirs is None:
|
|
|
|
dirs = []
|
|
|
|
if context_processors is None:
|
|
|
|
context_processors = []
|
|
|
|
if loaders is None:
|
|
|
|
loaders = ['django.template.loaders.filesystem.Loader']
|
|
|
|
if app_dirs:
|
|
|
|
loaders += ['django.template.loaders.app_directories.Loader']
|
2016-09-03 21:06:33 +08:00
|
|
|
if not debug:
|
|
|
|
loaders = [('django.template.loaders.cached.Loader', loaders)]
|
2014-11-14 22:48:27 +08:00
|
|
|
else:
|
|
|
|
if app_dirs:
|
|
|
|
raise ImproperlyConfigured(
|
2014-12-15 06:18:38 +08:00
|
|
|
"app_dirs must not be set when loaders is defined.")
|
2015-05-09 04:10:36 +08:00
|
|
|
if libraries is None:
|
|
|
|
libraries = {}
|
|
|
|
if builtins is None:
|
|
|
|
builtins = []
|
2014-12-15 06:18:38 +08:00
|
|
|
|
2014-11-14 22:48:27 +08:00
|
|
|
self.dirs = dirs
|
|
|
|
self.app_dirs = app_dirs
|
2015-11-08 17:06:07 +08:00
|
|
|
self.autoescape = autoescape
|
2014-11-14 22:48:27 +08:00
|
|
|
self.context_processors = context_processors
|
2014-11-23 04:33:40 +08:00
|
|
|
self.debug = debug
|
2014-11-14 22:48:27 +08:00
|
|
|
self.loaders = loaders
|
|
|
|
self.string_if_invalid = string_if_invalid
|
2014-11-21 04:27:56 +08:00
|
|
|
self.file_charset = file_charset
|
2015-05-09 04:10:36 +08:00
|
|
|
self.libraries = libraries
|
|
|
|
self.template_libraries = self.get_template_libraries(libraries)
|
|
|
|
self.builtins = self.default_builtins + builtins
|
|
|
|
self.template_builtins = self.get_template_builtins(self.builtins)
|
2014-11-14 22:48:27 +08:00
|
|
|
|
2021-05-28 08:45:40 +08:00
|
|
|
def __repr__(self):
|
|
|
|
return (
|
|
|
|
'<%s:%s app_dirs=%s%s debug=%s loaders=%s string_if_invalid=%s '
|
|
|
|
'file_charset=%s%s%s autoescape=%s>'
|
|
|
|
) % (
|
|
|
|
self.__class__.__qualname__,
|
|
|
|
'' if not self.dirs else ' dirs=%s' % repr(self.dirs),
|
|
|
|
self.app_dirs,
|
|
|
|
''
|
|
|
|
if not self.context_processors
|
|
|
|
else ' context_processors=%s' % repr(self.context_processors),
|
|
|
|
self.debug,
|
|
|
|
repr(self.loaders),
|
|
|
|
repr(self.string_if_invalid),
|
|
|
|
repr(self.file_charset),
|
|
|
|
'' if not self.libraries else ' libraries=%s' % repr(self.libraries),
|
|
|
|
'' if not self.builtins else ' builtins=%s' % repr(self.builtins),
|
|
|
|
repr(self.autoescape),
|
|
|
|
)
|
|
|
|
|
2014-11-28 04:30:35 +08:00
|
|
|
@staticmethod
|
2021-09-27 15:10:58 +08:00
|
|
|
@functools.lru_cache
|
2014-11-28 04:30:35 +08:00
|
|
|
def get_default():
|
|
|
|
"""
|
2017-03-11 04:46:37 +08:00
|
|
|
Return the first DjangoTemplates backend that's configured, or raise
|
|
|
|
ImproperlyConfigured if none are configured.
|
2014-11-28 04:30:35 +08:00
|
|
|
|
|
|
|
This is required for preserving historical APIs that rely on a
|
|
|
|
globally available, implicitly configured engine such as:
|
|
|
|
|
|
|
|
>>> from django.template import Context, Template
|
|
|
|
>>> template = Template("Hello {{ name }}!")
|
|
|
|
>>> context = Context({'name': "world"})
|
|
|
|
>>> template.render(context)
|
|
|
|
'Hello world!'
|
|
|
|
"""
|
2015-01-04 05:50:20 +08:00
|
|
|
# Since Engine is imported in django.template and since
|
|
|
|
# DjangoTemplates is a wrapper around this Engine class,
|
|
|
|
# local imports are required to avoid import loops.
|
|
|
|
from django.template import engines
|
2014-11-28 04:30:35 +08:00
|
|
|
from django.template.backends.django import DjangoTemplates
|
2017-03-11 04:46:37 +08:00
|
|
|
for engine in engines.all():
|
|
|
|
if isinstance(engine, DjangoTemplates):
|
|
|
|
return engine.engine
|
|
|
|
raise ImproperlyConfigured('No DjangoTemplates backend is configured.')
|
2014-11-14 22:48:27 +08:00
|
|
|
|
2014-11-21 05:34:59 +08:00
|
|
|
@cached_property
|
|
|
|
def template_context_processors(self):
|
|
|
|
context_processors = _builtin_context_processors
|
|
|
|
context_processors += tuple(self.context_processors)
|
|
|
|
return tuple(import_string(path) for path in context_processors)
|
|
|
|
|
2015-05-09 04:10:36 +08:00
|
|
|
def get_template_builtins(self, builtins):
|
|
|
|
return [import_library(x) for x in builtins]
|
|
|
|
|
|
|
|
def get_template_libraries(self, libraries):
|
|
|
|
loaded = {}
|
|
|
|
for name, path in libraries.items():
|
|
|
|
loaded[name] = import_library(path)
|
|
|
|
return loaded
|
|
|
|
|
2014-11-14 22:48:27 +08:00
|
|
|
@cached_property
|
|
|
|
def template_loaders(self):
|
2014-11-20 06:23:58 +08:00
|
|
|
return self.get_template_loaders(self.loaders)
|
|
|
|
|
|
|
|
def get_template_loaders(self, template_loaders):
|
|
|
|
loaders = []
|
|
|
|
for template_loader in template_loaders:
|
|
|
|
loader = self.find_template_loader(template_loader)
|
|
|
|
if loader is not None:
|
|
|
|
loaders.append(loader)
|
|
|
|
return loaders
|
|
|
|
|
|
|
|
def find_template_loader(self, loader):
|
|
|
|
if isinstance(loader, (tuple, list)):
|
2018-09-28 21:57:12 +08:00
|
|
|
loader, *args = loader
|
2014-11-20 06:23:58 +08:00
|
|
|
else:
|
|
|
|
args = []
|
|
|
|
|
2016-12-29 23:27:49 +08:00
|
|
|
if isinstance(loader, str):
|
2014-11-20 06:23:58 +08:00
|
|
|
loader_class = import_string(loader)
|
2015-09-05 20:37:44 +08:00
|
|
|
return loader_class(self, *args)
|
2014-11-20 06:23:58 +08:00
|
|
|
else:
|
|
|
|
raise ImproperlyConfigured(
|
|
|
|
"Invalid value in template loaders configuration: %r" % loader)
|
2014-11-14 22:48:27 +08:00
|
|
|
|
2015-03-04 05:48:26 +08:00
|
|
|
def find_template(self, name, dirs=None, skip=None):
|
|
|
|
tried = []
|
2014-11-14 22:48:27 +08:00
|
|
|
for loader in self.template_loaders:
|
2016-12-31 09:09:26 +08:00
|
|
|
try:
|
|
|
|
template = loader.get_template(name, skip=skip)
|
|
|
|
return template, template.origin
|
|
|
|
except TemplateDoesNotExist as e:
|
|
|
|
tried.extend(e.tried)
|
2015-03-04 05:48:26 +08:00
|
|
|
raise TemplateDoesNotExist(name, tried=tried)
|
2014-11-14 22:48:27 +08:00
|
|
|
|
2014-11-29 05:29:45 +08:00
|
|
|
def from_string(self, template_code):
|
2014-11-29 05:33:15 +08:00
|
|
|
"""
|
2017-01-25 04:36:36 +08:00
|
|
|
Return a compiled Template object for the given template code,
|
2014-11-29 05:33:15 +08:00
|
|
|
handling template inheritance recursively.
|
|
|
|
"""
|
2014-11-29 05:29:45 +08:00
|
|
|
return Template(template_code, engine=self)
|
2014-11-29 05:33:15 +08:00
|
|
|
|
2015-09-04 04:12:22 +08:00
|
|
|
def get_template(self, template_name):
|
2014-11-14 22:48:27 +08:00
|
|
|
"""
|
2017-01-25 04:36:36 +08:00
|
|
|
Return a compiled Template object for the given template name,
|
2014-11-14 22:48:27 +08:00
|
|
|
handling template inheritance recursively.
|
|
|
|
"""
|
2015-09-04 04:12:22 +08:00
|
|
|
template, origin = self.find_template(template_name)
|
2014-11-14 22:48:27 +08:00
|
|
|
if not hasattr(template, 'render'):
|
|
|
|
# template needs to be compiled
|
2014-11-29 05:29:45 +08:00
|
|
|
template = Template(template, origin, template_name, engine=self)
|
2014-11-14 22:48:27 +08:00
|
|
|
return template
|
|
|
|
|
2015-09-04 10:01:30 +08:00
|
|
|
def render_to_string(self, template_name, context=None):
|
2015-09-24 20:20:06 +08:00
|
|
|
"""
|
|
|
|
Render the template specified by template_name with the given context.
|
|
|
|
For use in Django's test suite.
|
|
|
|
"""
|
2014-11-14 22:48:27 +08:00
|
|
|
if isinstance(template_name, (list, tuple)):
|
2015-09-04 04:12:22 +08:00
|
|
|
t = self.select_template(template_name)
|
2014-11-14 22:48:27 +08:00
|
|
|
else:
|
2015-09-04 04:12:22 +08:00
|
|
|
t = self.get_template(template_name)
|
2015-09-04 10:01:30 +08:00
|
|
|
# Django < 1.8 accepted a Context in `context` even though that's
|
|
|
|
# unintended. Preserve this ability but don't rewrap `context`.
|
|
|
|
if isinstance(context, Context):
|
|
|
|
return t.render(context)
|
|
|
|
else:
|
2019-03-24 04:27:45 +08:00
|
|
|
return t.render(Context(context, autoescape=self.autoescape))
|
2014-11-14 22:48:27 +08:00
|
|
|
|
2015-09-04 04:12:22 +08:00
|
|
|
def select_template(self, template_name_list):
|
2014-11-14 22:48:27 +08:00
|
|
|
"""
|
2017-01-25 04:36:36 +08:00
|
|
|
Given a list of template names, return the first that can be loaded.
|
2014-11-14 22:48:27 +08:00
|
|
|
"""
|
|
|
|
if not template_name_list:
|
|
|
|
raise TemplateDoesNotExist("No template names provided")
|
|
|
|
not_found = []
|
|
|
|
for template_name in template_name_list:
|
|
|
|
try:
|
2015-09-04 04:12:22 +08:00
|
|
|
return self.get_template(template_name)
|
2014-11-14 22:48:27 +08:00
|
|
|
except TemplateDoesNotExist as exc:
|
|
|
|
if exc.args[0] not in not_found:
|
|
|
|
not_found.append(exc.args[0])
|
|
|
|
continue
|
|
|
|
# If we get here, none of the templates could be loaded
|
|
|
|
raise TemplateDoesNotExist(', '.join(not_found))
|