# coding: utf-8 """ Unit tests of template.py. """ import codecs import os import sys import unittest from .examples.simple import Simple from pystache import Renderer from pystache import TemplateSpec from pystache.common import TemplateNotFoundError from pystache.context import ContextStack, KeyNotFoundError from pystache.loader import Loader from pystache.tests.common import get_data_path, AssertStringMixin, AssertExceptionMixin from pystache.tests.data.views import SayHello def _make_renderer(): """ Return a default Renderer instance for testing purposes. """ renderer = Renderer(string_encoding='ascii', file_encoding='ascii') return renderer def mock_unicode(b, encoding=None): if encoding is None: encoding = 'ascii' u = str(b, encoding=encoding) return u.upper() class RendererInitTestCase(unittest.TestCase): """ Tests the Renderer.__init__() method. """ def test_partials__default(self): """ Test the default value. """ renderer = Renderer() self.assertTrue(renderer.partials is None) def test_partials(self): """ Test that the attribute is set correctly. """ renderer = Renderer(partials={'foo': 'bar'}) self.assertEqual(renderer.partials, {'foo': 'bar'}) def test_escape__default(self): escape = Renderer().escape self.assertEqual(escape(">"), ">") self.assertEqual(escape('"'), """) # Single quotes are escaped only in Python 3.2 and later. if sys.version_info < (3, 2): expected = "'" else: expected = ''' self.assertEqual(escape("'"), expected) def test_escape(self): escape = lambda s: "**" + s renderer = Renderer(escape=escape) self.assertEqual(renderer.escape("bar"), "**bar") def test_decode_errors__default(self): """ Check the default value. """ renderer = Renderer() self.assertEqual(renderer.decode_errors, 'strict') def test_decode_errors(self): """ Check that the constructor sets the attribute correctly. """ renderer = Renderer(decode_errors="foo") self.assertEqual(renderer.decode_errors, "foo") def test_file_encoding__default(self): """ Check the file_encoding default. """ renderer = Renderer() self.assertEqual(renderer.file_encoding, renderer.string_encoding) def test_file_encoding(self): """ Check that the file_encoding attribute is set correctly. """ renderer = Renderer(file_encoding='foo') self.assertEqual(renderer.file_encoding, 'foo') def test_file_extension__default(self): """ Check the file_extension default. """ renderer = Renderer() self.assertEqual(renderer.file_extension, 'mustache') def test_file_extension(self): """ Check that the file_encoding attribute is set correctly. """ renderer = Renderer(file_extension='foo') self.assertEqual(renderer.file_extension, 'foo') def test_missing_tags(self): """ Check that the missing_tags attribute is set correctly. """ renderer = Renderer(missing_tags='foo') self.assertEqual(renderer.missing_tags, 'foo') def test_missing_tags__default(self): """ Check the missing_tags default. """ renderer = Renderer() self.assertEqual(renderer.missing_tags, 'ignore') def test_search_dirs__default(self): """ Check the search_dirs default. """ renderer = Renderer() self.assertEqual(renderer.search_dirs, [os.curdir]) def test_search_dirs__string(self): """ Check that the search_dirs attribute is set correctly when a string. """ renderer = Renderer(search_dirs='foo') self.assertEqual(renderer.search_dirs, ['foo']) def test_search_dirs__list(self): """ Check that the search_dirs attribute is set correctly when a list. """ renderer = Renderer(search_dirs=['foo']) self.assertEqual(renderer.search_dirs, ['foo']) def test_string_encoding__default(self): """ Check the default value. """ renderer = Renderer() self.assertEqual(renderer.string_encoding, sys.getdefaultencoding()) def test_string_encoding(self): """ Check that the constructor sets the attribute correctly. """ renderer = Renderer(string_encoding="foo") self.assertEqual(renderer.string_encoding, "foo") class RendererTests(unittest.TestCase, AssertStringMixin): """Test the Renderer class.""" def _renderer(self): return Renderer() ## Test Renderer.unicode(). def test_unicode__string_encoding(self): """ Test that the string_encoding attribute is respected. """ renderer = self._renderer() b = "é".encode('utf-8') renderer.string_encoding = "ascii" self.assertRaises(UnicodeDecodeError, renderer.str, b) renderer.string_encoding = "utf-8" self.assertEqual(renderer.str(b), "é") def test_unicode__decode_errors(self): """ Test that the decode_errors attribute is respected. """ renderer = self._renderer() renderer.string_encoding = "ascii" b = "déf".encode('utf-8') renderer.decode_errors = "ignore" self.assertEqual(renderer.str(b), "df") renderer.decode_errors = "replace" # U+FFFD is the official Unicode replacement character. self.assertEqual(renderer.str(b), u'd\ufffd\ufffdf') ## Test the _make_loader() method. def test__make_loader__return_type(self): """ Test that _make_loader() returns a Loader. """ renderer = self._renderer() loader = renderer._make_loader() self.assertEqual(type(loader), Loader) def test__make_loader__attributes(self): """ Test that _make_loader() sets all attributes correctly.. """ unicode_ = lambda x: x renderer = self._renderer() renderer.file_encoding = 'enc' renderer.file_extension = 'ext' renderer.str = unicode_ loader = renderer._make_loader() self.assertEqual(loader.extension, 'ext') self.assertEqual(loader.file_encoding, 'enc') self.assertEqual(loader.to_unicode, unicode_) ## Test the render() method. def test_render__return_type(self): """ Check that render() returns a string of type unicode. """ renderer = self._renderer() rendered = renderer.render('foo') self.assertEqual(type(rendered), str) def test_render__unicode(self): renderer = self._renderer() actual = renderer.render('foo') self.assertEqual(actual, 'foo') def test_render__str(self): renderer = self._renderer() actual = renderer.render('foo') self.assertEqual(actual, 'foo') def test_render__non_ascii_character(self): renderer = self._renderer() actual = renderer.render('Poincaré') self.assertEqual(actual, 'Poincaré') def test_render__context(self): """ Test render(): passing a context. """ renderer = self._renderer() self.assertEqual(renderer.render('Hi {{person}}', {'person': 'Mom'}), 'Hi Mom') def test_render__context_and_kwargs(self): """ Test render(): passing a context and **kwargs. """ renderer = self._renderer() template = 'Hi {{person1}} and {{person2}}' self.assertEqual(renderer.render(template, {'person1': 'Mom'}, person2='Dad'), 'Hi Mom and Dad') def test_render__kwargs_and_no_context(self): """ Test render(): passing **kwargs and no context. """ renderer = self._renderer() self.assertEqual(renderer.render('Hi {{person}}', person='Mom'), 'Hi Mom') def test_render__context_and_kwargs__precedence(self): """ Test render(): **kwargs takes precedence over context. """ renderer = self._renderer() self.assertEqual(renderer.render('Hi {{person}}', {'person': 'Mom'}, person='Dad'), 'Hi Dad') def test_render__kwargs_does_not_modify_context(self): """ Test render(): passing **kwargs does not modify the passed context. """ context = {} renderer = self._renderer() renderer.render('Hi {{person}}', context=context, foo="bar") self.assertEqual(context, {}) def test_render__nonascii_template(self): """ Test passing a non-unicode template with non-ascii characters. """ renderer = _make_renderer() template = "déf".encode("utf-8") # Check that decode_errors and string_encoding are both respected. renderer.decode_errors = 'ignore' renderer.string_encoding = 'ascii' self.assertEqual(renderer.render(template), "df") renderer.string_encoding = 'utf_8' self.assertEqual(renderer.render(template), "déf") def test_make_resolve_partial(self): """ Test the _make_resolve_partial() method. """ renderer = Renderer() renderer.partials = {'foo': 'bar'} resolve_partial = renderer._make_resolve_partial() actual = resolve_partial('foo') self.assertEqual(actual, 'bar') self.assertEqual(type(actual), str, "RenderEngine requires that " "resolve_partial return unicode strings.") def test_make_resolve_partial__unicode(self): """ Test _make_resolve_partial(): that resolve_partial doesn't "double-decode" Unicode. """ renderer = Renderer() renderer.partials = {'partial': 'foo'} resolve_partial = renderer._make_resolve_partial() self.assertEqual(resolve_partial("partial"), "foo") # Now with a value that is already unicode. renderer.partials = {'partial': 'foo'} resolve_partial = renderer._make_resolve_partial() # If the next line failed, we would get the following error: # TypeError: decoding Unicode is not supported self.assertEqual(resolve_partial("partial"), "foo") def test_render_name(self): """Test the render_name() method.""" data_dir = get_data_path() renderer = Renderer(search_dirs=data_dir) actual = renderer.render_name("say_hello", to='foo') self.assertString(actual, "Hello, foo") def test_render_path(self): """ Test the render_path() method. """ renderer = Renderer() path = get_data_path('say_hello.mustache') actual = renderer.render_path(path, to='foo') self.assertEqual(actual, "Hello, foo") def test_render__object(self): """ Test rendering an object instance. """ renderer = Renderer() say_hello = SayHello() actual = renderer.render(say_hello) self.assertEqual('Hello, World', actual) actual = renderer.render(say_hello, to='Mars') self.assertEqual('Hello, Mars', actual) def test_render__template_spec(self): """ Test rendering a TemplateSpec instance. """ renderer = Renderer() class Spec(TemplateSpec): template = "hello, {{to}}" to = 'world' spec = Spec() actual = renderer.render(spec) self.assertString(actual, 'hello, world') def test_render__view(self): """ Test rendering a View instance. """ renderer = Renderer() view = Simple() actual = renderer.render(view) self.assertEqual('Hi pizza!', actual) def test_custom_string_coercion_via_assignment(self): """ Test that string coercion can be customized via attribute assignment. """ renderer = self._renderer() def to_str(val): if not val: return '' else: return str(val) self.assertEqual(renderer.render('{{value}}', value=None), 'None') renderer.str_coerce = to_str self.assertEqual(renderer.render('{{value}}', value=None), '') def test_custom_string_coercion_via_subclassing(self): """ Test that string coercion can be customized via subclassing. """ class MyRenderer(Renderer): def str_coerce(self, val): if not val: return '' else: return str(val) renderer1 = Renderer() renderer2 = MyRenderer() self.assertEqual(renderer1.render('{{value}}', value=None), 'None') self.assertEqual(renderer2.render('{{value}}', value=None), '') # By testing that Renderer.render() constructs the right RenderEngine, # we no longer need to exercise all rendering code paths through # the Renderer. It suffices to test rendering paths through the # RenderEngine for the same amount of code coverage. class Renderer_MakeRenderEngineTests(unittest.TestCase, AssertStringMixin, AssertExceptionMixin): """ Check the RenderEngine returned by Renderer._make_render_engine(). """ def _make_renderer(self): """ Return a default Renderer instance for testing purposes. """ return _make_renderer() ## Test the engine's resolve_partial attribute. def test__resolve_partial__returns_unicode(self): """ Check that resolve_partial returns unicode (and not a subclass). """ class MyUnicode(str): pass renderer = Renderer() renderer.string_encoding = 'ascii' renderer.partials = {'str': 'foo', 'subclass': MyUnicode('abc')} engine = renderer._make_render_engine() actual = engine.resolve_partial('str') self.assertEqual(actual, "foo") self.assertEqual(type(actual), str) # Check that unicode subclasses are not preserved. actual = engine.resolve_partial('subclass') self.assertEqual(actual, "abc") self.assertEqual(type(actual), str) def test__resolve_partial__not_found(self): """ Check that resolve_partial returns the empty string when a template is not found. """ renderer = Renderer() engine = renderer._make_render_engine() resolve_partial = engine.resolve_partial self.assertString(resolve_partial('foo'), '') def test__resolve_partial__not_found__missing_tags_strict(self): """ Check that resolve_partial provides a nice message when a template is not found. """ renderer = Renderer() renderer.missing_tags = 'strict' engine = renderer._make_render_engine() resolve_partial = engine.resolve_partial self.assertException(TemplateNotFoundError, "File 'foo.mustache' not found in dirs: ['.']", resolve_partial, "foo") def test__resolve_partial__not_found__partials_dict(self): """ Check that resolve_partial returns the empty string when a template is not found. """ renderer = Renderer() renderer.partials = {} engine = renderer._make_render_engine() resolve_partial = engine.resolve_partial self.assertString(resolve_partial('foo'), '') def test__resolve_partial__not_found__partials_dict__missing_tags_strict(self): """ Check that resolve_partial provides a nice message when a template is not found. """ renderer = Renderer() renderer.missing_tags = 'strict' renderer.partials = {} engine = renderer._make_render_engine() resolve_partial = engine.resolve_partial # Include dict directly since str(dict) is different in Python 2 and 3: # versus , respectively. self.assertException(TemplateNotFoundError, "Name 'foo' not found in partials: %s" % dict, resolve_partial, "foo") ## Test the engine's literal attribute. def test__literal__uses_renderer_unicode(self): """ Test that literal uses the renderer's unicode function. """ renderer = self._make_renderer() renderer.str = mock_unicode engine = renderer._make_render_engine() literal = engine.literal b = "foo".encode("ascii") self.assertEqual(literal(b), "FOO") def test__literal__handles_unicode(self): """ Test that literal doesn't try to "double decode" unicode. """ renderer = Renderer() renderer.string_encoding = 'ascii' engine = renderer._make_render_engine() literal = engine.literal self.assertEqual(literal("foo"), "foo") def test__literal__returns_unicode(self): """ Test that literal returns unicode (and not a subclass). """ renderer = Renderer() renderer.string_encoding = 'ascii' engine = renderer._make_render_engine() literal = engine.literal self.assertEqual(type(literal("foo")), str) class MyUnicode(str): pass s = MyUnicode("abc") self.assertEqual(type(s), MyUnicode) self.assertTrue(isinstance(s, str)) self.assertEqual(type(literal(s)), str) ## Test the engine's escape attribute. def test__escape__uses_renderer_escape(self): """ Test that escape uses the renderer's escape function. """ renderer = Renderer() renderer.escape = lambda s: "**" + s engine = renderer._make_render_engine() escape = engine.escape self.assertEqual(escape("foo"), "**foo") def test__escape__uses_renderer_unicode(self): """ Test that escape uses the renderer's unicode function. """ renderer = Renderer() renderer.str = mock_unicode engine = renderer._make_render_engine() escape = engine.escape b = "foo".encode('ascii') self.assertEqual(escape(b), "FOO") def test__escape__has_access_to_original_unicode_subclass(self): """ Test that escape receives strings with the unicode subclass intact. """ renderer = Renderer() renderer.escape = lambda s: str(type(s).__name__) engine = renderer._make_render_engine() escape = engine.escape class MyUnicode(str): pass self.assertEqual(escape("foo".encode('ascii')), str.__name__) self.assertEqual(escape("foo"), str.__name__) self.assertEqual(escape(MyUnicode("foo")), MyUnicode.__name__) def test__escape__returns_unicode(self): """ Test that literal returns unicode (and not a subclass). """ renderer = Renderer() renderer.string_encoding = 'ascii' engine = renderer._make_render_engine() escape = engine.escape self.assertEqual(type(escape("foo")), str) # Check that literal doesn't preserve unicode subclasses. class MyUnicode(str): pass s = MyUnicode("abc") self.assertEqual(type(s), MyUnicode) self.assertTrue(isinstance(s, str)) self.assertEqual(type(escape(s)), str) ## Test the missing_tags attribute. def test__missing_tags__unknown_value(self): """ Check missing_tags attribute: setting an unknown value. """ renderer = Renderer() renderer.missing_tags = 'foo' self.assertException(Exception, "Unsupported 'missing_tags' value: 'foo'", renderer._make_render_engine) ## Test the engine's resolve_context attribute. def test__resolve_context(self): """ Check resolve_context(): default arguments. """ renderer = Renderer() engine = renderer._make_render_engine() stack = ContextStack({'foo': 'bar'}) self.assertEqual('bar', engine.resolve_context(stack, 'foo')) self.assertString('', engine.resolve_context(stack, 'missing')) def test__resolve_context__missing_tags_strict(self): """ Check resolve_context(): missing_tags 'strict'. """ renderer = Renderer() renderer.missing_tags = 'strict' engine = renderer._make_render_engine() stack = ContextStack({'foo': 'bar'}) self.assertEqual('bar', engine.resolve_context(stack, 'foo')) self.assertException(KeyNotFoundError, "Key 'missing' not found: first part", engine.resolve_context, stack, 'missing')