Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

referencing variables from defaults/main.yml in ansible role

I have a role setup as follows

roles/test/task/main.yml

- name: Generate people files
  template: src=test.j2 dest=/tmp/{{ item.name }}.cfg
  loop: "{{people}}"

roles/test/template/test.j2

First Var: {{ item.var1 }}
Second Var: {{ item.var2 }}

roles/test/vars/main.yml

---
people:
        - name: TheSimpsons
          var1: homer
          var2: simpson

        - name: StarWars
          var1: han
          var2: solo

roles/test/defaults/main.yml

people:
   - var2: skywalker

my playbook

 - hosts: localhost
   roles:
    - test

When I run my playbook everything works as expect. I get two new files in /tmp with the correct text. However if I remove this var2 line from my vars/main.yml file...

          var2: solo

I would expect the var2 value from my defaults/main.yml to show up in the output, but all I get is this error

failed: [localhost] (item={u'var1': u'han', u'name': u'StarWars'}) => {
    "changed": false,
    "item": {
        "name": "StarWars",
        "var1": "han"
    },
    "msg": "AnsibleUndefinedVariable: 'dict object' has no attribute 'var2'"
}

I have tried formatting my defaults/main.yml about 10 different ways but get the same error each time.

If I setup a test that doesn't loop and defaults/main.yml and vars/main.yml are flat "key: value" pairs I can get it to pull values from defaults/main/yml just fine.

Something about the looping I'm just not getting. What am I doing wrong?

like image 425
pizzaguy39 Avatar asked Oct 27 '25 23:10

pizzaguy39


1 Answers

There are more options on how to combine the lists' items. For example, rename the defaults, e.g.

people_defaults:
   - var2: skywalker

Rename the vars too and create the list

people_vars:
  - name: TheSimpsons
    var1: homer
    var2: simpson
  - name: StarWars
    var1: than
people: "{{ people_defaults|product(people_vars)|map('combine')|list }}"

gives

people:
  - name: TheSimpsons
    var1: homer
    var2: simpson
  - name: StarWars
    var1: han
    var2: skywalker

Details of the role

shell> tree roles/test/
roles/test/
├── defaults
│   └── main.yml
├── tasks
│   └── main.yml
└── vars
    └── main.yml

3 directories, 3 files
shell> cat roles/test/defaults/main.yml 
people_defaults:
  - var2: skywalker
shell> cat roles/test/vars/main.yml 
people_vars:
  - name: TheSimpsons
    var1: homer
    var2: simpson
  - name: StarWars
    var1: han
people: "{{ people_defaults|product(people_vars)|map('combine')|list }}"
shell> cat roles/test/tasks/main.yml 
- debug:
    var: people

Error: combine expects dictionaries

If you run the code on Ansible 2.9 or older you'll see the error

{"msg": "|combine expects dictionaries, got ({'var2': 'skywalker'}, {'name': 'TheSimpsons', 'var1': 'homer', 'var2': 'simpson'})"}

Fix the problem by mapping the tuples to the lists. This fix is upwards compatible.

people: "{{ people_defaults|product(people_vars)|map('list')|map('combine')|list }}"

Q: "Combination of the lists does not preserve the hierarchy of variables."

A: You're combining dictionaries. A dictionary(mapping) in YAML is an unordered set of key/value node pairs. See Mapping. The sorting is up to you. See Ansible list not ordered.

like image 184
Vladimir Botka Avatar answered Oct 29 '25 14:10

Vladimir Botka



Donate For Us

If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!