salt/tests/unit/templates/jinja_test.py

150 lines
5.4 KiB
Python
Raw Normal View History

import os
import tempfile
2011-12-29 23:58:00 +00:00
from jinja2 import Environment
from salt.utils.jinja import SaltCacheLoader
from salt.utils.templates import render_jinja_tmpl
2011-12-29 23:58:00 +00:00
from saltunittest import TestCase
TEMPLATES_DIR = os.path.dirname(os.path.abspath(__file__))
2011-12-29 23:58:00 +00:00
2012-05-29 16:40:20 +00:00
2011-12-29 23:58:00 +00:00
class MockFileClient(object):
'''
Does not download files but records any file request for testing
'''
def __init__(self, loader=None):
2012-05-29 16:40:20 +00:00
if loader:
loader._file_client = self
2011-12-29 23:58:00 +00:00
self.requests = []
2012-05-29 16:40:20 +00:00
2011-12-29 23:58:00 +00:00
def get_file(self, template, dest='', makedirs=False, env='base'):
self.requests.append({
'path': template,
2011-12-29 23:58:00 +00:00
'dest': dest,
'makedirs': makedirs,
'env': env
})
2012-05-29 16:40:20 +00:00
class TestSaltCacheLoader(TestCase):
2011-12-29 23:58:00 +00:00
def test_searchpath(self):
'''
The searchpath is based on the cachedir option and the env parameter
'''
tmp = tempfile.gettempdir()
loader = SaltCacheLoader({'cachedir': tmp}, env='test')
assert loader.searchpath == os.path.join(tmp, 'files', 'test')
2012-05-29 16:40:20 +00:00
2011-12-29 23:58:00 +00:00
def test_mockclient(self):
'''
2012-06-30 20:10:34 +00:00
A MockFileClient is used that records all file requests normally sent
to the master.
2011-12-29 23:58:00 +00:00
'''
loader = SaltCacheLoader({'cachedir': TEMPLATES_DIR}, 'test')
fc = MockFileClient(loader)
res = loader.get_source(None, 'hello_simple')
assert len(res) == 3
2012-06-25 00:10:54 +00:00
# res[0] on Windows is unicode and use os.linesep so it works cross OS
self.assertEqual(str(res[0]), 'world' + os.linesep)
tmpl_dir = os.path.join(TEMPLATES_DIR, 'files', 'test', 'hello_simple')
self.assertEqual(res[1], tmpl_dir)
2011-12-29 23:58:00 +00:00
assert res[2](), "Template up to date?"
assert len(fc.requests)
self.assertEqual(fc.requests[0]['path'], 'salt://hello_simple')
2012-05-29 16:40:20 +00:00
2011-12-29 23:58:00 +00:00
def get_test_env(self):
'''
Setup a simple jinja test environment
'''
loader = SaltCacheLoader({'cachedir': TEMPLATES_DIR}, 'test')
fc = MockFileClient(loader)
jinja = Environment(loader=loader)
return fc, jinja
2012-05-29 16:40:20 +00:00
2011-12-29 23:58:00 +00:00
def test_import(self):
'''
You can import and use macros from other files
'''
fc, jinja = self.get_test_env()
result = jinja.get_template('hello_import').render()
self.assertEqual(result, 'Hey world !a b !')
assert len(fc.requests) == 2
self.assertEqual(fc.requests[0]['path'], 'salt://hello_import')
self.assertEqual(fc.requests[1]['path'], 'salt://macro')
2012-05-29 16:40:20 +00:00
2011-12-29 23:58:00 +00:00
def test_include(self):
'''
You can also include a template that imports and uses macros
'''
fc, jinja = self.get_test_env()
result = jinja.get_template('hello_include').render()
self.assertEqual(result, 'Hey world !a b !')
assert len(fc.requests) == 3
self.assertEqual(fc.requests[0]['path'], 'salt://hello_include')
self.assertEqual(fc.requests[1]['path'], 'salt://hello_import')
self.assertEqual(fc.requests[2]['path'], 'salt://macro')
2012-05-29 16:40:20 +00:00
2011-12-29 23:58:00 +00:00
def test_include_context(self):
'''
Context variables are passes to the included template by default.
'''
_, jinja = self.get_test_env()
result = jinja.get_template('hello_include').render(a='Hi', b='Salt')
self.assertEqual(result, 'Hey world !Hi Salt !')
2012-05-29 16:40:20 +00:00
class TestGetTemplate(TestCase):
def __init__(self, *args, **kws):
TestCase.__init__(self, *args, **kws)
self.local_opts = {
'cachedir': TEMPLATES_DIR,
'file_client': 'local',
'file_roots': {
'other': os.path.join(TEMPLATES_DIR, 'files', 'test')
}
}
2011-12-29 23:58:00 +00:00
def test_fallback(self):
'''
A Template with a filesystem loader is returned as fallback
2011-12-29 23:58:00 +00:00
if the file is not contained in the searchpath
'''
2012-11-14 03:32:55 +00:00
fn_ = os.path.join(TEMPLATES_DIR, 'files', 'test', 'hello_simple')
with open(fn_) as fp_:
out = render_jinja_tmpl(
fp_.read(),
dict(opts=self.local_opts, env='other'))
self.assertEqual(out, 'world')
2012-05-29 16:40:20 +00:00
2011-12-29 23:58:00 +00:00
def test_fallback_noloader(self):
'''
A Template with a filesystem loader is returned as fallback
if the file is not contained in the searchpath
2011-12-29 23:58:00 +00:00
'''
filename = os.path.join(TEMPLATES_DIR, 'files', 'test', 'hello_import')
out = render_jinja_tmpl(
open(filename).read(),
dict(opts=self.local_opts, env='other'))
self.assertEqual(out, 'Hey world !a b !')
2012-05-29 16:40:20 +00:00
2011-12-29 23:58:00 +00:00
def test_env(self):
'''
If the template is within the searchpath it can
import, include and extend other templates.
The initial template is expected to be already cached
get_template does not request it from the master again.
'''
fc = MockFileClient()
# monkey patch file client
_fc = SaltCacheLoader.file_client
SaltCacheLoader.file_client = lambda loader: fc
filename = os.path.join(TEMPLATES_DIR, 'files', 'test', 'hello_import')
out = render_jinja_tmpl(
open(filename).read(),
dict(opts={'cachedir': TEMPLATES_DIR, 'file_client': 'remote'},
a='Hi', b='Salt', env='test'))
self.assertEqual(out, 'Hey world !Hi Salt !')
2011-12-29 23:58:00 +00:00
self.assertEqual(fc.requests[0]['path'], 'salt://macro')
SaltCacheLoader.file_client = _fc