allow references to names of variables in with_items without needing to surround them with Jinja2 '{{' delimeters
This commit is contained in:
parent
9ac25bb8f6
commit
f0b21dcc0f
5 changed files with 61 additions and 40 deletions
|
@ -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'
|
||||||
|
|
|
@ -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)
|
||||||
|
|
|
@ -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 = []
|
||||||
|
|
|
@ -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
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -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:
|
||||||
|
|
Loading…
Add table
Reference in a new issue