Ansible playbook 只添加角色依赖变量

Ansible playbook only add role dependency variables

所有文件是

# roles/a/defaults/main.yml
test: Role A

# roles/a/tasks/main.yml
- name: Things I dont want to run
  debug: msg="Not expected"


# hosts
host-1 host-1 ansible_connection=local test=Override


# test.yml
- hosts: all
  roles:
    - a # I only need the variable
  tasks:
    - name: action
      debug: var=test # Expected Override

我该怎么做?

我想过这个

roles/
    a-var # only include defaults variable definition
    a # depends on a-var
    b # depends on a-var, but do not run role a's task

有点麻烦,但我认为这可能是唯一的解决方法。

为什么我想要这个是我不想让用户关心太多文件(group_vars,host_vars,extra_vars),他们不知道ansible,我必须尽可能简单地部署我们的系统,他们唯一需要关心的就是库存文件,就像这样

host-1
host-2

[a]
host-1 port=1234

[b]
host-2 # use default port defined in role

[all:vars]
install_path=/opt # override the default path

但是加载变量的顺序是

这对这个组织来说很难,因为我不能这样做,var 文件将胜过库存配置。

- host: b
  var_files:
    # I hope this is just default and not take over inventory config
    - roles/a/defaults/main.yml 
  tasks:
    - debug: var=install_path # become default not custom's

我看不出你的计划有什么问题。这确实很麻烦,但会起作用。

另一个想法是使用条件:

roles:
  - role: a
    when: False

这不会执行角色 a 内的任何任务,但它的变量和默认值将被评估,稍后可用于其他角色。

这里的丑陋之处在于,Ansible 会将条件传递给角色的每个任务。它仍然会在输出中显示为跳过的任务。


但只是为了确保您了解 default() 过滤器:

您可以在模板或任务中执行类似的操作:

{{ port | default(42) }}

因此,如果在清单文件中定义了端口,它将被使用。如果未定义,则为 42。

您甚至可以将它与 set_fact 任务结合起来。

- set_fact:
    port: "{{ port | default(42) }}"

这是完成我认为是您的目标的最干净的方法。

如果您有更多需要默认值的变量,您可以在一个任务中全部完成:

- set_fact:
    port: "{{ port | default(42) }}"
    install_path: "{{ install_path | default('/opt') }}"