mirror of
				https://github.com/django/django.git
				synced 2025-10-24 14:16:09 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			236 lines
		
	
	
		
			8.5 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			236 lines
		
	
	
		
			8.5 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| """
 | |
| Test cases for the template loaders
 | |
| 
 | |
| Note: This test requires setuptools!
 | |
| """
 | |
| 
 | |
| from django.conf import settings
 | |
| 
 | |
| if __name__ == '__main__':
 | |
|     settings.configure()
 | |
| 
 | |
| import os.path
 | |
| import sys
 | |
| import types
 | |
| import unittest
 | |
| 
 | |
| try:
 | |
|     import pkg_resources
 | |
| except ImportError:
 | |
|     pkg_resources = None
 | |
| 
 | |
| 
 | |
| from django.template import TemplateDoesNotExist, Context
 | |
| from django.template.loaders.eggs import Loader as EggLoader
 | |
| from django.template import loader
 | |
| from django.test import TestCase, override_settings
 | |
| from django.utils import six
 | |
| from django.utils._os import upath
 | |
| from django.utils.six import StringIO
 | |
| 
 | |
| 
 | |
| # Mock classes and objects for pkg_resources functions.
 | |
| class MockLoader(object):
 | |
|     pass
 | |
| 
 | |
| 
 | |
| def create_egg(name, resources):
 | |
|     """
 | |
|     Creates a mock egg with a list of resources.
 | |
| 
 | |
|     name: The name of the module.
 | |
|     resources: A dictionary of resources. Keys are the names and values the data.
 | |
|     """
 | |
|     egg = types.ModuleType(name)
 | |
|     egg.__loader__ = MockLoader()
 | |
|     egg.__path__ = ['/some/bogus/path/']
 | |
|     egg.__file__ = '/some/bogus/path/__init__.pyc'
 | |
|     egg._resources = resources
 | |
|     sys.modules[name] = egg
 | |
| 
 | |
| 
 | |
| @unittest.skipUnless(pkg_resources, 'setuptools is not installed')
 | |
| class EggLoaderTest(TestCase):
 | |
|     def setUp(self):
 | |
|         # Defined here b/c at module scope we may not have pkg_resources
 | |
|         class MockProvider(pkg_resources.NullProvider):
 | |
|             def __init__(self, module):
 | |
|                 pkg_resources.NullProvider.__init__(self, module)
 | |
|                 self.module = module
 | |
| 
 | |
|             def _has(self, path):
 | |
|                 return path in self.module._resources
 | |
| 
 | |
|             def _isdir(self, path):
 | |
|                 return False
 | |
| 
 | |
|             def get_resource_stream(self, manager, resource_name):
 | |
|                 return self.module._resources[resource_name]
 | |
| 
 | |
|             def _get(self, path):
 | |
|                 return self.module._resources[path].read()
 | |
| 
 | |
|             def _fn(self, base, resource_name):
 | |
|                 return os.path.normcase(resource_name)
 | |
| 
 | |
|         pkg_resources._provider_factories[MockLoader] = MockProvider
 | |
| 
 | |
|         self.empty_egg = create_egg("egg_empty", {})
 | |
|         self.egg_1 = create_egg("egg_1", {
 | |
|             os.path.normcase('templates/y.html'): StringIO("y"),
 | |
|             os.path.normcase('templates/x.txt'): StringIO("x"),
 | |
|         })
 | |
| 
 | |
|     @override_settings(INSTALLED_APPS=['egg_empty'])
 | |
|     def test_empty(self):
 | |
|         "Loading any template on an empty egg should fail"
 | |
|         egg_loader = EggLoader()
 | |
|         self.assertRaises(TemplateDoesNotExist, egg_loader.load_template_source, "not-existing.html")
 | |
| 
 | |
|     @override_settings(INSTALLED_APPS=['egg_1'])
 | |
|     def test_non_existing(self):
 | |
|         "Template loading fails if the template is not in the egg"
 | |
|         egg_loader = EggLoader()
 | |
|         self.assertRaises(TemplateDoesNotExist, egg_loader.load_template_source, "not-existing.html")
 | |
| 
 | |
|     @override_settings(INSTALLED_APPS=['egg_1'])
 | |
|     def test_existing(self):
 | |
|         "A template can be loaded from an egg"
 | |
|         egg_loader = EggLoader()
 | |
|         contents, template_name = egg_loader.load_template_source("y.html")
 | |
|         self.assertEqual(contents, "y")
 | |
|         self.assertEqual(template_name, "egg:egg_1:templates/y.html")
 | |
| 
 | |
|     def test_not_installed(self):
 | |
|         "Loading an existent template from an egg not included in any app should fail"
 | |
|         egg_loader = EggLoader()
 | |
|         self.assertRaises(TemplateDoesNotExist, egg_loader.load_template_source, "y.html")
 | |
| 
 | |
| 
 | |
| @override_settings(
 | |
|     TEMPLATE_LOADERS=(
 | |
|         ('django.template.loaders.cached.Loader', (
 | |
|             'django.template.loaders.filesystem.Loader',
 | |
|         )),
 | |
|     )
 | |
| )
 | |
| class CachedLoader(TestCase):
 | |
|     def test_templatedir_caching(self):
 | |
|         "Check that the template directories form part of the template cache key. Refs #13573"
 | |
|         # Retrieve a template specifying a template directory to check
 | |
|         t1, name = loader.find_template('test.html', (os.path.join(os.path.dirname(upath(__file__)), 'templates', 'first'),))
 | |
|         # Now retrieve the same template name, but from a different directory
 | |
|         t2, name = loader.find_template('test.html', (os.path.join(os.path.dirname(upath(__file__)), 'templates', 'second'),))
 | |
| 
 | |
|         # The two templates should not have the same content
 | |
|         self.assertNotEqual(t1.render(Context({})), t2.render(Context({})))
 | |
| 
 | |
|     def test_missing_template_is_cached(self):
 | |
|         "#19949 -- Check that the missing template is cached."
 | |
|         template_loader = loader.find_template_loader(settings.TEMPLATE_LOADERS[0])
 | |
|         # Empty cache, which may be filled from previous tests.
 | |
|         template_loader.reset()
 | |
|         # Check that 'missing.html' isn't already in cache before 'missing.html' is loaded
 | |
|         self.assertRaises(KeyError, lambda: template_loader.template_cache["missing.html"])
 | |
|         # Try to load it, it should fail
 | |
|         self.assertRaises(TemplateDoesNotExist, template_loader.load_template, "missing.html")
 | |
|         # Verify that the fact that the missing template, which hasn't been found, has actually
 | |
|         # been cached:
 | |
|         self.assertEqual(template_loader.template_cache.get("missing.html"),
 | |
|                          TemplateDoesNotExist,
 | |
|                          "Cached template loader doesn't cache file lookup misses. It should.")
 | |
| 
 | |
| 
 | |
| @override_settings(
 | |
|     TEMPLATE_DIRS=(
 | |
|         os.path.join(os.path.dirname(upath(__file__)), 'templates'),
 | |
|     )
 | |
| )
 | |
| class RenderToStringTest(TestCase):
 | |
|     def test_basic(self):
 | |
|         self.assertEqual(loader.render_to_string('test_context.html'), 'obj:')
 | |
| 
 | |
|     def test_basic_context(self):
 | |
|         self.assertEqual(loader.render_to_string('test_context.html',
 | |
|                                                  {'obj': 'test'}), 'obj:test')
 | |
| 
 | |
|     def test_existing_context_kept_clean(self):
 | |
|         context = Context({'obj': 'before'})
 | |
|         output = loader.render_to_string('test_context.html', {'obj': 'after'},
 | |
|                                          context_instance=context)
 | |
|         self.assertEqual(output, 'obj:after')
 | |
|         self.assertEqual(context['obj'], 'before')
 | |
| 
 | |
|     def test_empty_list(self):
 | |
|         six.assertRaisesRegex(self, TemplateDoesNotExist,
 | |
|             'No template names provided$',
 | |
|             loader.render_to_string, [])
 | |
| 
 | |
|     def test_select_templates_from_empty_list(self):
 | |
|         six.assertRaisesRegex(self, TemplateDoesNotExist,
 | |
|             'No template names provided$',
 | |
|             loader.select_template, [])
 | |
| 
 | |
|     def test_no_empty_dict_pushed_to_stack(self):
 | |
|         """
 | |
|         No empty dict should be pushed to the context stack when render_to_string
 | |
|         is called without any argument (#21741).
 | |
|         """
 | |
| 
 | |
|         # The stack should have a length of 1, corresponding to the builtins
 | |
|         self.assertEqual('1',
 | |
|             loader.render_to_string('test_context_stack.html').strip())
 | |
|         self.assertEqual('1',
 | |
|             loader.render_to_string('test_context_stack.html', context_instance=Context()).strip())
 | |
| 
 | |
| 
 | |
| class TemplateDirsOverrideTest(unittest.TestCase):
 | |
| 
 | |
|     dirs_tuple = (os.path.join(os.path.dirname(upath(__file__)), 'other_templates'),)
 | |
|     dirs_list = list(dirs_tuple)
 | |
|     dirs_iter = (dirs_tuple, dirs_list)
 | |
| 
 | |
|     def test_render_to_string(self):
 | |
|         for dirs in self.dirs_iter:
 | |
|             self.assertEqual(loader.render_to_string('test_dirs.html', dirs=dirs), 'spam eggs\n')
 | |
| 
 | |
|     def test_get_template(self):
 | |
|         for dirs in self.dirs_iter:
 | |
|             template = loader.get_template('test_dirs.html', dirs=dirs)
 | |
|             self.assertEqual(template.render(Context({})), 'spam eggs\n')
 | |
| 
 | |
|     def test_select_template(self):
 | |
|         for dirs in self.dirs_iter:
 | |
|             template = loader.select_template(['test_dirs.html'], dirs=dirs)
 | |
|             self.assertEqual(template.render(Context({})), 'spam eggs\n')
 | |
| 
 | |
| 
 | |
| @override_settings(
 | |
|     TEMPLATE_LOADERS=(
 | |
|         ('django.template.loaders.cached.Loader', (
 | |
|             'django.template.loaders.filesystem.Loader',
 | |
|             'django.template.loaders.app_directories.Loader',
 | |
|         )),
 | |
|     )
 | |
| )
 | |
| class PriorityCacheLoader(TestCase):
 | |
|     def test_basic(self):
 | |
|         """
 | |
|         Check that the order of template loader works. Refs #21460.
 | |
|         """
 | |
|         t1, name = loader.find_template('priority/foo.html')
 | |
|         self.assertEqual(t1.render(Context({})), 'priority\n')
 | |
| 
 | |
| 
 | |
| @override_settings(
 | |
|     TEMPLATE_LOADERS=('django.template.loaders.filesystem.Loader',
 | |
|                       'django.template.loaders.app_directories.Loader',),
 | |
| )
 | |
| class PriorityLoader(TestCase):
 | |
|     def test_basic(self):
 | |
|         """
 | |
|         Check that the order of template loader works. Refs #21460.
 | |
|         """
 | |
|         t1, name = loader.find_template('priority/foo.html')
 | |
|         self.assertEqual(t1.render(Context({})), 'priority\n')
 |