allow references to names of variables in with_items without needing to surround them with Jinja2 '{{' delimeters

This commit is contained in:
Michael DeHaan 2013-04-10 18:42:54 -04:00
parent 9ac25bb8f6
commit f0b21dcc0f
5 changed files with 61 additions and 40 deletions

View file

@ -7,16 +7,25 @@
- hosts: all - hosts: all
tasks: vars:
- shell: echo 'hello {{ ansible_hostname.upper() }}' a_list:
- a
- b
- c
- shell: echo 'match' tasks:
- shell: echo hello {{ ansible_hostname.upper() }}
- shell: echo match
when: 2 == 2 when: 2 == 2
- shell: echo 'no match' - shell: echo no match
when: 2 == 2 + 1 when: 2 == 2 + 1
- shell: echo '{{ ansible_os_family }}' - shell: echo {{ ansible_os_family }}
- shell: echo {{ item }}
with_items: a_list
- shell: echo 'RedHat' - shell: echo 'RedHat'
when: ansible_os_family == 'RedHat' when: ansible_os_family == 'RedHat'

View file

@ -16,6 +16,7 @@
# along with Ansible. If not, see <http://www.gnu.org/licenses/>. # along with Ansible. If not, see <http://www.gnu.org/licenses/>.
from ansible.utils import safe_eval from ansible.utils import safe_eval
import ansible.utils.template as template
def flatten(terms): def flatten(terms):
ret = [] ret = []
@ -28,12 +29,20 @@ def flatten(terms):
class LookupModule(object): class LookupModule(object):
def __init__(self, **kwargs): def __init__(self, basedir=None, **kwargs):
pass self.basedir = basedir
def run(self, terms, **kwargs): def run(self, terms, inject=None, **kwargs):
if isinstance(terms, basestring): if isinstance(terms, basestring):
# somewhat did:
# with_items: alist
# OR
# with_items: {{ alist }}
if not '{' in terms and not '[' in terms:
terms = '{{ %s }}' % terms
terms = template.template(self.basedir, terms, inject)
if '{' or '[' in terms: if '{' or '[' in terms:
# Jinja2 already evaluated a variable to a list.
# Jinja2-ified list needs to be converted back to a real type # Jinja2-ified list needs to be converted back to a real type
# TODO: something a bit less heavy than eval # TODO: something a bit less heavy than eval
terms = safe_eval(terms) terms = safe_eval(terms)

View file

@ -15,7 +15,9 @@
# You should have received a copy of the GNU General Public License # You should have received a copy of the GNU General Public License
# along with Ansible. If not, see <http://www.gnu.org/licenses/>. # along with Ansible. If not, see <http://www.gnu.org/licenses/>.
import ansible.utils.template as template
from ansible.utils import safe_eval from ansible.utils import safe_eval
import ansible.errors as errors
def flatten(terms): def flatten(terms):
ret = [] ret = []
@ -37,16 +39,30 @@ def combine(a,b):
class LookupModule(object): class LookupModule(object):
def __init__(self, **kwargs): def __init__(self, basedir=None, **kwargs):
pass self.basedir = basedir
def run(self, terms, inject=None, **kwargs):
# this code is common with 'items.py' consider moving to utils if we need it again
if isinstance(terms, basestring):
# someone did:
# with_items: alist
# OR
# with_items: {{ alist }}
if not '{' in terms and not '[' in terms:
terms = '{{ %s }}' % terms
terms = template.template(self.basedir, terms, inject)
if '{' or '[' in terms:
# Jinja2 already evaluated a variable to a list.
# Jinja2-ified list needs to be converted back to a real type
# TODO: something a bit less heavy than eval
terms = safe_eval(terms)
def run(self, terms, **kwargs):
if '{' or '[' in terms:
# Jinja2-ified list needs to be converted back to a real type
# TODO: something a bit less heavy than eval
terms = safe_eval(terms)
if not isinstance(terms, list): if not isinstance(terms, list):
raise errors.AnsibleError("a list is required for with_nested") raise errors.AnsibleError("a list is required for with_nested")
my_list = terms[:] my_list = terms[:]
my_list.reverse() my_list.reverse()
result = [] result = []

View file

@ -254,28 +254,6 @@ def parse_json(raw_data):
return { "failed" : True, "parsed" : False, "msg" : orig_data } return { "failed" : True, "parsed" : False, "msg" : orig_data }
return results return results
def preprocess_yaml(data):
# allow the following casual user error:
# bar: {{ baz }}
# instead of bar: '{{ baz }}'
# (commander-API should auto-quote these in similar ways on save/load)
new_lines = []
lines = data.split("\n")
for line in lines:
if line.find("{{") != -1 and line.find(":") != -1:
tokens = line.split(":",1)
if tokens[1].strip().startswith("{{"):
new_line = "%s: '%s'" % (tokens[0], tokens[1])
else:
new_line = line
new_lines.append(new_line)
else:
new_lines.append(line)
result = "\n".join(new_lines)
return result
def parse_yaml(data): def parse_yaml(data):
''' convert a yaml string to a data structure ''' ''' convert a yaml string to a data structure '''
return yaml.safe_load(data) return yaml.safe_load(data)
@ -310,7 +288,6 @@ def parse_yaml_from_file(path):
try: try:
data = file(path).read() data = file(path).read()
data = preprocess_yaml(data)
return parse_yaml(data) return parse_yaml(data)
except IOError: except IOError:
raise errors.AnsibleError("file not found: %s" % path) raise errors.AnsibleError("file not found: %s" % path)
@ -724,7 +701,12 @@ def safe_eval(str):
# do not allow imports # do not allow imports
if re.search(r'import \w+', str): if re.search(r'import \w+', str):
return str return str
return eval(str) try:
return eval(str)
except Exception, e:
return str

View file

@ -435,7 +435,12 @@ def template_from_string(basedir, data, vars):
raise errors.AnsibleError("recursive loop detected in template string: %s" % data) raise errors.AnsibleError("recursive loop detected in template string: %s" % data)
else: else:
return data return data
def test_foo():
return 'test_foo!'
t.globals['test_foo'] = test_foo
res = jinja2.utils.concat(t.root_render_func(t.new_context(_jinja2_vars(basedir, vars, t.globals), shared=True))) res = jinja2.utils.concat(t.root_render_func(t.new_context(_jinja2_vars(basedir, vars, t.globals), shared=True)))
return res return res
except jinja2.exceptions.UndefinedError: except jinja2.exceptions.UndefinedError: