test_loader.py 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325
  1. # -*- coding: utf-8 -*-
  2. import os
  3. import shutil
  4. import sys
  5. import tempfile
  6. import time
  7. import weakref
  8. import pytest
  9. from jinja2 import Environment
  10. from jinja2 import loaders
  11. from jinja2._compat import PY2
  12. from jinja2._compat import PYPY
  13. from jinja2.exceptions import TemplateNotFound
  14. from jinja2.loaders import split_template_path
  15. import yatest.common as yc
  16. class TestLoaders(object):
  17. def test_dict_loader(self, dict_loader):
  18. env = Environment(loader=dict_loader)
  19. tmpl = env.get_template("justdict.html")
  20. assert tmpl.render().strip() == "FOO"
  21. pytest.raises(TemplateNotFound, env.get_template, "missing.html")
  22. def test_package_loader(self, package_loader):
  23. env = Environment(loader=package_loader)
  24. tmpl = env.get_template("test.html")
  25. assert tmpl.render().strip() == "BAR"
  26. pytest.raises(TemplateNotFound, env.get_template, "missing.html")
  27. def test_filesystem_loader_overlapping_names(self, filesystem_loader):
  28. res = os.path.dirname(filesystem_loader.searchpath[0])
  29. t2_dir = os.path.join(res, "templates2")
  30. # Make "foo" show up before "foo/test.html".
  31. filesystem_loader.searchpath.insert(0, t2_dir)
  32. e = Environment(loader=filesystem_loader)
  33. e.get_template("foo")
  34. # This would raise NotADirectoryError if "t2/foo" wasn't skipped.
  35. e.get_template("foo/test.html")
  36. def test_choice_loader(self, choice_loader):
  37. env = Environment(loader=choice_loader)
  38. tmpl = env.get_template("justdict.html")
  39. assert tmpl.render().strip() == "FOO"
  40. tmpl = env.get_template("test.html")
  41. assert tmpl.render().strip() == "BAR"
  42. pytest.raises(TemplateNotFound, env.get_template, "missing.html")
  43. def test_function_loader(self, function_loader):
  44. env = Environment(loader=function_loader)
  45. tmpl = env.get_template("justfunction.html")
  46. assert tmpl.render().strip() == "FOO"
  47. pytest.raises(TemplateNotFound, env.get_template, "missing.html")
  48. def test_prefix_loader(self, prefix_loader):
  49. env = Environment(loader=prefix_loader)
  50. tmpl = env.get_template("a/test.html")
  51. assert tmpl.render().strip() == "BAR"
  52. tmpl = env.get_template("b/justdict.html")
  53. assert tmpl.render().strip() == "FOO"
  54. pytest.raises(TemplateNotFound, env.get_template, "missing")
  55. def test_caching(self):
  56. changed = False
  57. class TestLoader(loaders.BaseLoader):
  58. def get_source(self, environment, template):
  59. return u"foo", None, lambda: not changed
  60. env = Environment(loader=TestLoader(), cache_size=-1)
  61. tmpl = env.get_template("template")
  62. assert tmpl is env.get_template("template")
  63. changed = True
  64. assert tmpl is not env.get_template("template")
  65. changed = False
  66. def test_no_cache(self):
  67. mapping = {"foo": "one"}
  68. env = Environment(loader=loaders.DictLoader(mapping), cache_size=0)
  69. assert env.get_template("foo") is not env.get_template("foo")
  70. def test_limited_size_cache(self):
  71. mapping = {"one": "foo", "two": "bar", "three": "baz"}
  72. loader = loaders.DictLoader(mapping)
  73. env = Environment(loader=loader, cache_size=2)
  74. t1 = env.get_template("one")
  75. t2 = env.get_template("two")
  76. assert t2 is env.get_template("two")
  77. assert t1 is env.get_template("one")
  78. env.get_template("three")
  79. loader_ref = weakref.ref(loader)
  80. assert (loader_ref, "one") in env.cache
  81. assert (loader_ref, "two") not in env.cache
  82. assert (loader_ref, "three") in env.cache
  83. def test_cache_loader_change(self):
  84. loader1 = loaders.DictLoader({"foo": "one"})
  85. loader2 = loaders.DictLoader({"foo": "two"})
  86. env = Environment(loader=loader1, cache_size=2)
  87. assert env.get_template("foo").render() == "one"
  88. env.loader = loader2
  89. assert env.get_template("foo").render() == "two"
  90. def test_dict_loader_cache_invalidates(self):
  91. mapping = {"foo": "one"}
  92. env = Environment(loader=loaders.DictLoader(mapping))
  93. assert env.get_template("foo").render() == "one"
  94. mapping["foo"] = "two"
  95. assert env.get_template("foo").render() == "two"
  96. def test_split_template_path(self):
  97. assert split_template_path("foo/bar") == ["foo", "bar"]
  98. assert split_template_path("./foo/bar") == ["foo", "bar"]
  99. pytest.raises(TemplateNotFound, split_template_path, "../foo")
  100. class TestFileSystemLoader(object):
  101. searchpath = os.path.join(
  102. yc.test_source_path(), "res", "templates"
  103. )
  104. @staticmethod
  105. def _test_common(env):
  106. tmpl = env.get_template("test.html")
  107. assert tmpl.render().strip() == "BAR"
  108. tmpl = env.get_template("foo/test.html")
  109. assert tmpl.render().strip() == "FOO"
  110. pytest.raises(TemplateNotFound, env.get_template, "missing.html")
  111. def test_searchpath_as_str(self):
  112. filesystem_loader = loaders.FileSystemLoader(self.searchpath)
  113. env = Environment(loader=filesystem_loader)
  114. self._test_common(env)
  115. @pytest.mark.skipif(PY2, reason="pathlib is not available in Python 2")
  116. def test_searchpath_as_pathlib(self):
  117. import pathlib
  118. searchpath = pathlib.Path(self.searchpath)
  119. filesystem_loader = loaders.FileSystemLoader(searchpath)
  120. env = Environment(loader=filesystem_loader)
  121. self._test_common(env)
  122. @pytest.mark.skipif(PY2, reason="pathlib is not available in Python 2")
  123. def test_searchpath_as_list_including_pathlib(self):
  124. import pathlib
  125. searchpath = pathlib.Path(self.searchpath)
  126. filesystem_loader = loaders.FileSystemLoader(["/tmp/templates", searchpath])
  127. env = Environment(loader=filesystem_loader)
  128. self._test_common(env)
  129. @pytest.mark.skip("Arcadia read only")
  130. def test_caches_template_based_on_mtime(self):
  131. filesystem_loader = loaders.FileSystemLoader(self.searchpath)
  132. env = Environment(loader=filesystem_loader)
  133. tmpl1 = env.get_template("test.html")
  134. tmpl2 = env.get_template("test.html")
  135. assert tmpl1 is tmpl2
  136. os.utime(os.path.join(self.searchpath, "test.html"), (time.time(), time.time()))
  137. tmpl3 = env.get_template("test.html")
  138. assert tmpl1 is not tmpl3
  139. @pytest.mark.parametrize(
  140. ("encoding", "expect"),
  141. [
  142. ("utf-8", u"文字化け"),
  143. ("iso-8859-1", u"æ\x96\x87\xe5\xad\x97\xe5\x8c\x96\xe3\x81\x91"),
  144. ],
  145. )
  146. def test_uses_specified_encoding(self, encoding, expect):
  147. loader = loaders.FileSystemLoader(self.searchpath, encoding=encoding)
  148. e = Environment(loader=loader)
  149. t = e.get_template("mojibake.txt")
  150. assert t.render() == expect
  151. class TestModuleLoader(object):
  152. archive = None
  153. def compile_down(self, prefix_loader, zip="deflated", py_compile=False):
  154. log = []
  155. self.reg_env = Environment(loader=prefix_loader)
  156. if zip is not None:
  157. fd, self.archive = tempfile.mkstemp(suffix=".zip")
  158. os.close(fd)
  159. else:
  160. self.archive = tempfile.mkdtemp()
  161. self.reg_env.compile_templates(
  162. self.archive, zip=zip, log_function=log.append, py_compile=py_compile
  163. )
  164. self.mod_env = Environment(loader=loaders.ModuleLoader(self.archive))
  165. return "".join(log)
  166. def teardown(self):
  167. if hasattr(self, "mod_env"):
  168. if os.path.isfile(self.archive):
  169. os.remove(self.archive)
  170. else:
  171. shutil.rmtree(self.archive)
  172. self.archive = None
  173. def test_log(self, prefix_loader):
  174. log = self.compile_down(prefix_loader)
  175. assert (
  176. 'Compiled "a/foo/test.html" as '
  177. "tmpl_a790caf9d669e39ea4d280d597ec891c4ef0404a" in log
  178. )
  179. assert "Finished compiling templates" in log
  180. assert (
  181. 'Could not compile "a/syntaxerror.html": '
  182. "Encountered unknown tag 'endif'" in log
  183. )
  184. def _test_common(self):
  185. tmpl1 = self.reg_env.get_template("a/test.html")
  186. tmpl2 = self.mod_env.get_template("a/test.html")
  187. assert tmpl1.render() == tmpl2.render()
  188. tmpl1 = self.reg_env.get_template("b/justdict.html")
  189. tmpl2 = self.mod_env.get_template("b/justdict.html")
  190. assert tmpl1.render() == tmpl2.render()
  191. def test_deflated_zip_compile(self, prefix_loader):
  192. self.compile_down(prefix_loader, zip="deflated")
  193. self._test_common()
  194. def test_stored_zip_compile(self, prefix_loader):
  195. self.compile_down(prefix_loader, zip="stored")
  196. self._test_common()
  197. def test_filesystem_compile(self, prefix_loader):
  198. self.compile_down(prefix_loader, zip=None)
  199. self._test_common()
  200. def test_weak_references(self, prefix_loader):
  201. self.compile_down(prefix_loader)
  202. self.mod_env.get_template("a/test.html")
  203. key = loaders.ModuleLoader.get_template_key("a/test.html")
  204. name = self.mod_env.loader.module.__name__
  205. assert hasattr(self.mod_env.loader.module, key)
  206. assert name in sys.modules
  207. # unset all, ensure the module is gone from sys.modules
  208. self.mod_env = None
  209. try:
  210. import gc
  211. gc.collect()
  212. except BaseException:
  213. pass
  214. assert name not in sys.modules
  215. # This test only makes sense on non-pypy python 2
  216. @pytest.mark.skipif(
  217. not (PY2 and not PYPY), reason="This test only makes sense on non-pypy python 2"
  218. )
  219. def test_byte_compilation(self, prefix_loader):
  220. log = self.compile_down(prefix_loader, py_compile=True)
  221. assert 'Byte-compiled "a/test.html"' in log
  222. self.mod_env.get_template("a/test.html")
  223. mod = self.mod_env.loader.module.tmpl_3c4ddf650c1a73df961a6d3d2ce2752f1b8fd490
  224. assert mod.__file__.endswith(".pyc")
  225. def test_choice_loader(self, prefix_loader):
  226. self.compile_down(prefix_loader)
  227. self.mod_env.loader = loaders.ChoiceLoader(
  228. [self.mod_env.loader, loaders.DictLoader({"DICT_SOURCE": "DICT_TEMPLATE"})]
  229. )
  230. tmpl1 = self.mod_env.get_template("a/test.html")
  231. assert tmpl1.render() == "BAR"
  232. tmpl2 = self.mod_env.get_template("DICT_SOURCE")
  233. assert tmpl2.render() == "DICT_TEMPLATE"
  234. def test_prefix_loader(self, prefix_loader):
  235. self.compile_down(prefix_loader)
  236. self.mod_env.loader = loaders.PrefixLoader(
  237. {
  238. "MOD": self.mod_env.loader,
  239. "DICT": loaders.DictLoader({"test.html": "DICT_TEMPLATE"}),
  240. }
  241. )
  242. tmpl1 = self.mod_env.get_template("MOD/a/test.html")
  243. assert tmpl1.render() == "BAR"
  244. tmpl2 = self.mod_env.get_template("DICT/test.html")
  245. assert tmpl2.render() == "DICT_TEMPLATE"
  246. @pytest.mark.skipif(PY2, reason="pathlib is not available in Python 2")
  247. def test_path_as_pathlib(self, prefix_loader):
  248. self.compile_down(prefix_loader)
  249. mod_path = self.mod_env.loader.module.__path__[0]
  250. import pathlib
  251. mod_loader = loaders.ModuleLoader(pathlib.Path(mod_path))
  252. self.mod_env = Environment(loader=mod_loader)
  253. self._test_common()
  254. @pytest.mark.skipif(PY2, reason="pathlib is not available in Python 2")
  255. def test_supports_pathlib_in_list_of_paths(self, prefix_loader):
  256. self.compile_down(prefix_loader)
  257. mod_path = self.mod_env.loader.module.__path__[0]
  258. import pathlib
  259. mod_loader = loaders.ModuleLoader([pathlib.Path(mod_path), "/tmp/templates"])
  260. self.mod_env = Environment(loader=mod_loader)
  261. self._test_common()