15
votes

Does anyone know how to put playbooks into folders, but share the same roles, group_vars, and other stuff typically located at the root dir?

Here's what I'd like to have:

root_dir:
- group_vars
- roles
- inventory
- playbooks
  - my_playbook.yml
- site.yml
- deploy.yml

Our root dir is getting pretty big now and I'd like to split out some playbooks into their own folder (shown as playbooks/ above). An identical tiny playbook fails to run when inside a directory (say, playbooks/) vs at the root dir, because it doesn't grab stuff from group_vars.

I can partially work around this, and run a playbook inside my playbooks/ directory:

- hosts: host_group
  sudo: true
  gather_facts: false
  vars_files:
  - ../group_vars/all/main.yml

This picks up the vars defined in main.yml. However, it's not clear to me if this would add group variables defined in group_vars/, as opposed to the explicitly specificed ../group_vars/.

Thanks!

5

5 Answers

7
votes

Ansible will pick up group_vars without stating path explicitly:

Here is the example directory structure in /tmp/ansible:

/tmp/ansible
├── group_vars
│   └── test_group.yml
├── inventory
├── playbooks
│   └── foo.yml
└── site.yml

Inventory file:

[test_group]
localhost

Main Playbook site.yml:

# site.yml
---
- include: playbooks/foo.yml 

Secondary Playbook foo.yml:

---
- hosts: localhost

  tasks:

      - debug: var=foo

Group Variables test_group.yml:

foo: bar

Here are all the ways to execute the secondary playbook:

  • From root folder using relative path /tmp/ansible:

    ansible-playbook -i inventory playbooks/foo.yml -c local
    
    PLAY [localhost] **************************************************************
    
    GATHERING FACTS ***************************************************************
    ok: [localhost]
    
    TASK: [debug var=foo] *********************************************************
    ok: [localhost] => {
        "var": {
            "foo": "bar"
        }
    }
    
    PLAY RECAP ********************************************************************
    localhost                  : ok=2    changed=0    unreachable=0    failed=0
    
  • From Playbooks subfolder with inventory in parent path /tmp/ansible/playbooks:

    ansible-playbook -i ../inventory foo.yml -c local
    
    PLAY [localhost] **************************************************************
    
    GATHERING FACTS ***************************************************************
    ok: [localhost]
    
    TASK: [debug var=foo] *********************************************************
    ok: [localhost] => {
        "var": {
            "foo": "bar"
        }
    }
    
    PLAY RECAP ********************************************************************
    localhost                  : ok=2    changed=0    unreachable=0    failed=0
    

As we can see in the above examples, ansible-playbook will look for variables based on the path of inventory file or folder were ansible was executed from. Playbooks can be separated without additional effort.

5
votes

Gregory Shulov's answer works if you don't use roles. If you use roles, though, your subfolder playbooks won't be able to reference them normally.

If you don't have too many files, I would recommend separating them "subfolder" files by prepending them with an underscore. That helped me separate some building block plays from the real playbooks I expect people to run.

5
votes

The group/host vars are relative to wherever your inventory is defined. I had a similar issue to your when I started. Here are the steps I took

  1. move your inventory (static or dynamic) into a separate directory. ie $PROJECT_HOME/inventory/base_inventory. Ansible will automatically parse all files in that directory and add to inventory. This always be overridden later. eg ansible -i inventory/other_hosts

  2. update ansible.cfg to point to the new inventory directory. ie. hostfile = $PROJECT_HOME/inventory

  3. move your groups_vars and host_vars into the new inventory directory. ie $PROJECT_HOME/inventory/group_vars

Now you don't need to worry about relative pathing Also if you want to user your roles from anywhere in your source tree update ansible.cfg and set the roles_path to the base project path. ie roles_path = roles:$PROJECT_HOME/roles

1
votes

I ran into the same problem.

ln -s ../roles playbooks/roles

just symlinking the roles/vars folder solved it.

# site.yml
---
- include: "playbooks/appserver.yml"
- include: "playbooks/gitlab-runner.yml"

or include them direct from the playbook dir

# playbooks/site.yml
---
- include: "appserver.yml"
- include: "gitlab-runner.yml"
1
votes

You can define this in ansible.cfg this will solve your problem

Example:

```

roles_path = /etc/ansible/roles ```