条件语句
在剧本中,您可能希望根据事实(有关远程系统的数据)、变量或先前任务的结果来执行不同的任务或有不同的目标。您可能希望某些变量的值取决于其他变量的值。或者您可能希望根据主机是否匹配其他条件来创建额外的主机组。您可以使用条件语句来完成所有这些操作。
Ansible 在条件语句中使用 Jinja2 测试 和 过滤器。Ansible 支持所有标准测试和过滤器,并添加了一些独特的测试和过滤器。
注意
在 Ansible 中,有许多选项可以控制执行流程。您可以在 https://jinja.flask.org.cn/en/latest/templates/#comparisons 中找到更多支持条件语句的示例。
使用 when
的基本条件语句
最简单的条件语句适用于单个任务。创建任务,然后添加一个 when
语句来应用测试。 when
子句是一个原始的 Jinja2 表达式,没有双花括号(请参见 group_by_module)。当您运行任务或剧本时,Ansible 会针对所有主机评估测试。在测试通过(返回 True 值)的任何主机上,Ansible 都会运行该任务。例如,如果您在多台机器上安装 mysql,其中一些机器启用了 SELinux,您可能需要一个任务来配置 SELinux 以允许 mysql 运行。您只希望该任务在启用了 SELinux 的机器上运行。
tasks:
- name: Configure SELinux to start mysql on any port
ansible.posix.seboolean:
name: mysql_connect_any
state: true
persistent: true
when: ansible_selinux.status == "enabled"
# all variables can be used directly in conditionals without double curly braces
基于 ansible_facts 的条件语句
通常,您希望根据事实来执行或跳过任务。事实是各个主机的属性,包括 IP 地址、操作系统、文件系统的状态等等。使用基于事实的条件语句
您可以在操作系统是特定版本时安装特定软件包。
您可以在具有内部 IP 地址的主机上跳过配置防火墙。
您可以在文件系统即将满时执行清理任务。
请参见 常用事实,了解条件语句中经常出现的各事实的列表。并非所有事实都存在于所有主机上。例如,以下示例中使用的 “lsb_major_release” 事实仅在目标主机上安装了 lsb_release package
时才存在。要查看您的系统上可用的事实,请将调试任务添加到您的剧本中。
- name: Show facts available on the system
ansible.builtin.debug:
var: ansible_facts
以下是一个基于事实的条件语句示例。
tasks:
- name: Shut down Debian flavored systems
ansible.builtin.command: /sbin/shutdown -t now
when: ansible_facts['os_family'] == "Debian"
如果您有多个条件,可以使用括号将它们分组。
tasks:
- name: Shut down CentOS 6 and Debian 7 systems
ansible.builtin.command: /sbin/shutdown -t now
when: (ansible_facts['distribution'] == "CentOS" and ansible_facts['distribution_major_version'] == "6") or
(ansible_facts['distribution'] == "Debian" and ansible_facts['distribution_major_version'] == "7")
您可以使用 逻辑运算符 来组合条件。当您有多个条件,并且所有条件都需要为真(即逻辑 and
)时,可以将它们指定为列表。
tasks:
- name: Shut down CentOS 6 systems
ansible.builtin.command: /sbin/shutdown -t now
when:
- ansible_facts['distribution'] == "CentOS"
- ansible_facts['distribution_major_version'] == "6"
如果事实或变量是字符串,并且您需要对其进行数学比较,请使用过滤器来确保 Ansible 将该值读取为整数。
tasks:
- ansible.builtin.shell: echo "only on Red Hat 6, derivatives, and later"
when: ansible_facts['os_family'] == "RedHat" and ansible_facts['lsb']['major_release'] | int >= 6
您可以将 Ansible 事实存储为变量,用于条件逻辑,如下例所示。
tasks:
- name: Get the CPU temperature
set_fact:
temperature: "{{ ansible_facts['cpu_temperature'] }}"
- name: Restart the system if the temperature is too high
when: temperature | float > 90
shell: "reboot"
基于注册变量的条件语句
在剧本中,您通常希望根据先前任务的结果来执行或跳过任务。例如,您可能希望在先前任务升级服务后配置服务。要创建基于注册变量的条件语句,请执行以下操作。
将先前任务的结果注册为变量。
创建一个基于注册变量的条件测试。
您使用 register
关键字创建注册变量的名称。注册变量始终包含创建它的任务的状态,以及任务生成的所有输出。您可以在模板和操作行以及条件 when
语句中使用注册变量。您可以使用 variable.stdout
访问注册变量的字符串内容。例如。
- name: Test play
hosts: all
tasks:
- name: Register a variable
ansible.builtin.shell: cat /etc/motd
register: motd_contents
- name: Use the variable in conditional statement
ansible.builtin.shell: echo "motd contains the word hi"
when: motd_contents.stdout.find('hi') != -1
如果变量是列表,您可以在任务循环中使用注册结果。如果变量不是列表,您可以使用 stdout_lines
或 variable.stdout.split()
将其转换为列表。您还可以按其他字段拆分行。
- name: Registered variable usage as a loop list
hosts: all
tasks:
- name: Retrieve the list of home directories
ansible.builtin.command: ls /home
register: home_dirs
- name: Add home dirs to the backup spooler
ansible.builtin.file:
path: /mnt/bkspool/{{ item }}
src: /home/{{ item }}
state: link
loop: "{{ home_dirs.stdout_lines }}"
# same as loop: "{{ home_dirs.stdout.split() }}"
注册变量的字符串内容可能为空。如果您希望仅在注册变量的 stdout 为空的主机上运行另一个任务,请检查注册变量的字符串内容是否为空。
- name: check registered variable for emptiness
hosts: all
tasks:
- name: List contents of directory
ansible.builtin.command: ls mydir
register: contents
- name: Check contents for emptiness
ansible.builtin.debug:
msg: "Directory is empty"
when: contents.stdout == ""
Ansible 始终在注册变量中为每个主机注册一些内容,即使在任务失败或 Ansible 由于不满足条件而跳过任务的主机上也是如此。要在此类主机上运行后续任务,请查询注册变量以获取 is skipped
(而不是 “undefined” 或 “default”)。有关更多信息,请参见 注册变量。以下是一些基于任务成功或失败的条件语句示例。请记住,如果您希望 Ansible 在发生错误时继续在主机上执行,请忽略错误。
tasks:
- name: Register a variable, ignore errors and continue
ansible.builtin.command: /bin/false
register: result
ignore_errors: true
- name: Run only if the task that registered the "result" variable fails
ansible.builtin.command: /bin/something
when: result is failed
- name: Run only if the task that registered the "result" variable succeeds
ansible.builtin.command: /bin/something_else
when: result is succeeded
- name: Run only if the task that registered the "result" variable is skipped
ansible.builtin.command: /bin/still/something_else
when: result is skipped
- name: Run only if the task that registered the "result" variable changed something.
ansible.builtin.command: /bin/still/something_else
when: result is changed
注意
较旧版本的 Ansible 使用 success
和 fail
,但 succeeded
和 failed
使用正确的时态。所有这些选项现在都有效。
基于变量的条件语句
您还可以创建基于在剧本或清单中定义的变量的条件语句。由于条件语句需要布尔值输入(测试必须评估为 True 才能触发条件),因此您必须对非布尔值变量(例如,包含 “yes”、“on”、“1” 或 “true” 等内容的字符串变量)应用 | bool
过滤器。您可以像这样定义变量。
vars:
epic: true
monumental: "yes"
使用上面的变量,Ansible 将运行以下任务之一,并跳过另一个任务。
tasks:
- name: Run the command if "epic" or "monumental" is true
ansible.builtin.shell: echo "This certainly is epic!"
when: epic or monumental | bool
- name: Run the command if "epic" is false
ansible.builtin.shell: echo "This certainly isn't epic!"
when: not epic
如果所需的变量尚未设置,您可以使用 Jinja2 的 defined 测试来跳过或失败。例如
tasks:
- name: Run the command if "foo" is defined
ansible.builtin.shell: echo "I've got '{{ foo }}' and am not afraid to use it!"
when: foo is defined
- name: Fail if "bar" is undefined
ansible.builtin.fail: msg="Bailing out. This play requires 'bar'"
when: bar is undefined
这在结合 vars
文件的条件导入时特别有用(见下文)。如示例所示,您不需要使用 {{ }} 来在条件语句中使用变量,因为这些变量已经是隐含的。
在循环中使用条件语句
如果您将 when
语句与 loop 结合使用,Ansible 会为每个项目分别处理条件语句。这是设计使然,因此您可以在循环中的某些项目上执行任务,并在其他项目上跳过它。例如
tasks:
- name: Run with items greater than 5
ansible.builtin.command: echo {{ item }}
loop: [ 0, 2, 4, 6, 8, 10 ]
when: item > 5
如果您需要在循环变量未定义时跳过整个任务,请使用 |default 过滤器提供一个空迭代器。例如,在循环遍历列表时
- name: Skip the whole task when a loop variable is undefined
ansible.builtin.command: echo {{ item }}
loop: "{{ mylist|default([]) }}"
when: item > 5
您可以在循环遍历字典时做同样的事情
- name: The same as above using a dict
ansible.builtin.command: echo {{ item.key }}
loop: "{{ query('dict', mydict|default({})) }}"
when: item.value > 5
加载自定义事实
您可以提供自己的事实,如 您是否需要开发模块? 中所述。要运行它们,只需在任务列表的顶部调用您自己的自定义事实收集模块,然后在以后的任务中可以访问那里返回的变量
tasks:
- name: Gather site specific fact data
action: site_facts
- name: Use a custom fact
ansible.builtin.command: /usr/bin/thingy
when: my_custom_fact_just_retrieved_from_the_remote_system == '1234'
条件语句与重用
您可以将条件语句与可重用任务文件、剧本或角色一起使用。Ansible 以不同的方式执行这些条件语句,用于动态重用(包含)和静态重用(导入)。有关 Ansible 中重用的更多信息,请参阅 重用 Ansible 资源。
条件语句与导入
当您向导入语句添加条件语句时,Ansible 会将条件应用于导入文件中所有任务。此行为等效于 标签继承:将标签添加到多个任务。Ansible 将条件应用于每个任务,并分别评估每个任务。例如,如果您想定义并显示一个以前未定义的变量,您可能有一个名为 main.yml
的剧本和一个名为 other_tasks.yml
的任务文件
# all tasks within an imported file inherit the condition from the import statement
# main.yml
- hosts: all
tasks:
- import_tasks: other_tasks.yml # note "import"
when: x is not defined
# other_tasks.yml
- name: Set a variable
ansible.builtin.set_fact:
x: foo
- name: Print a variable
ansible.builtin.debug:
var: x
Ansible 在执行时将此扩展到等效于
- name: Set a variable if not defined
ansible.builtin.set_fact:
x: foo
when: x is not defined
# this task sets a value for x
- name: Do the task if "x" is not defined
ansible.builtin.debug:
var: x
when: x is not defined
# Ansible skips this task, because x is now defined
如果 x
最初定义,则这两个任务都按预期跳过。但是,如果 x
最初未定义,则调试任务将跳过,因为条件语句对每个导入的任务都进行了评估。条件语句将对 set_fact
任务评估为 true
,这将定义变量并使 debug
条件语句评估为 false
。
如果这不是您想要的行为,请使用 include_*
语句仅将条件应用于该语句本身。
# using a conditional on include_* only applies to the include task itself
# main.yml
- hosts: all
tasks:
- include_tasks: other_tasks.yml # note "include"
when: x is not defined
现在,如果 x
最初未定义,则调试任务将不会跳过,因为条件语句在包含时进行评估,并且不适用于单个任务。
您也可以将条件应用于 import_playbook
以及其他 import_*
语句。当您使用此方法时,Ansible 会为每个不符合条件的主机上的每个任务返回一条“跳过”消息,从而创建重复的输出。在许多情况下,group_by 模块 可以成为完成相同目标的更简化的方式;请参阅 处理操作系统和发行版的差异。
条件语句与包含
当您在 include_*
语句上使用条件语句时,条件语句仅应用于包含任务本身,而不应用于包含文件中的任何其他任务。与上面用于导入条件语句的示例形成对比,请查看相同的剧本和任务文件,但使用包含而不是导入
# Includes let you reuse a file to define a variable when it is not already defined
# main.yml
- include_tasks: other_tasks.yml
when: x is not defined
# other_tasks.yml
- name: Set a variable
ansible.builtin.set_fact:
x: foo
- name: Print a variable
ansible.builtin.debug:
var: x
Ansible 在执行时将此扩展到等效于
# main.yml
- include_tasks: other_tasks.yml
when: x is not defined
# if condition is met, Ansible includes other_tasks.yml
# other_tasks.yml
- name: Set a variable
ansible.builtin.set_fact:
x: foo
# no condition applied to this task, Ansible sets the value of x to foo
- name: Print a variable
ansible.builtin.debug:
var: x
# no condition applied to this task, Ansible prints the debug statement
通过使用 include_tasks
而不是 import_tasks
,other_tasks.yml
中的两个任务将按预期执行。有关 include
与 import
之间差异的更多信息,请参阅 重用 Ansible 资源。
条件语句与角色
有三种方法可以将条件应用于角色
通过将
when
语句放在roles
关键字下,将相同的条件语句或条件语句添加到角色中的所有任务。请参阅本节中的示例。通过将
when
语句放在剧本中的静态import_role
上,将相同的条件语句或条件语句添加到角色中的所有任务。向角色本身内的单个任务或块添加一个或多个条件语句。这是唯一允许您根据
when
语句选择或跳过角色中某些任务的方法。要选择或跳过角色中的任务,您必须在单个任务或块上设置条件语句,在剧本中使用动态include_role
,并将条件语句或条件语句添加到包含语句中。当您使用此方法时,Ansible 会将条件应用于包含语句本身以及角色中也具有该when
语句的任何任务。
当您使用 roles
关键字将角色静态地整合到剧本中时,Ansible 会将您定义的条件语句添加到角色中的所有任务。例如
- hosts: webservers
roles:
- role: debian_stock_config
when: ansible_facts['os_family'] == 'Debian'
根据事实选择变量、文件或模板
有时,主机的事实决定了您要为某些变量使用的值,甚至决定了您要为该主机选择的文件或模板。例如,包的名称在 CentOS 和 Debian 上是不同的。常见服务的配置文件在不同的操作系统风格和版本上也是不同的。要根据主机的事实加载不同的变量文件、模板或其他文件
将您的变量文件、模板或文件命名为与区分它们的事实相匹配
使用基于该事实的变量为每个主机选择正确的变量文件、模板或文件
Ansible 将变量与任务分开,使您的剧本不会变成带有嵌套条件语句的任意代码。这种方法会导致更简化且可审计的配置规则,因为要跟踪的决策点更少。
根据事实选择变量文件
您可以创建一个在多个平台和操作系统版本上运行的剧本,只需将变量值放在变量文件中并有条件地导入它们。如果您想在某些 CentOS 和某些 Debian 服务器上安装 Apache,请创建具有 YAML 键和值的变量文件。例如
---
# for vars/RedHat.yml
apache: httpd
somethingelse: 42
然后,根据您在剧本中收集到的主机事实导入这些变量文件
---
- hosts: webservers
remote_user: root
vars_files:
- "vars/common.yml"
- [ "vars/{{ ansible_facts['os_family'] }}.yml", "vars/os_defaults.yml" ]
tasks:
- name: Make sure apache is started
ansible.builtin.service:
name: '{{ apache }}'
state: started
Ansible 收集 webservers 组中主机的 facts,然后将变量“ansible_facts[‘os_family’]”插入文件名列表中。如果您有使用 Red Hat 操作系统(例如 CentOS)的主机,Ansible 会查找“vars/RedHat.yml”。如果该文件不存在,Ansible 会尝试加载“vars/os_defaults.yml”。对于 Debian 主机,Ansible 首先会查找“vars/Debian.yml”,然后才会回退到“vars/os_defaults.yml”。如果列表中没有找到任何文件,Ansible 会引发错误。
根据事实选择文件和模板
当不同的操作系统风格或版本需要不同的配置文件或模板时,您可以使用相同的方法。根据分配给每个主机的变量选择合适的文件或模板。这种方法通常比将大量条件语句放在单个模板中以涵盖多个操作系统或包版本更简洁。
例如,您可以将配置文件模板化,使其在 CentOS 和 Debian 之间存在很大差异
- name: Template a file
ansible.builtin.template:
src: "{{ item }}"
dest: /etc/myapp/foo.conf
loop: "{{ query('first_found', { 'files': myfiles, 'paths': mypaths}) }}"
vars:
myfiles:
- "{{ ansible_facts['distribution'] }}.conf"
- default.conf
mypaths: ['search_location_one/somedir/', '/opt/other_location/somedir/']
调试条件语句
如果您的条件语句 when
的行为与预期不符,您可以添加一个 debug
语句来确定条件语句是否评估为 true
或 false
。条件语句中意外行为的常见原因是将整数作为字符串或将字符串作为整数进行测试。要调试条件语句,请将整个语句作为 var:
值添加到 debug
任务中。然后,Ansible 会显示测试以及语句的评估方式。例如,以下是一组任务和示例输出
- name: check value of return code
ansible.builtin.debug:
var: bar_status.rc
- name: check test for rc value as string
ansible.builtin.debug:
var: bar_status.rc == "127"
- name: check test for rc value as integer
ansible.builtin.debug:
var: bar_status.rc == 127
TASK [check value of return code] *********************************************************************************
ok: [foo-1] => {
"bar_status.rc": "127"
}
TASK [check test for rc value as string] **************************************************************************
ok: [foo-1] => {
"bar_status.rc == \"127\"": false
}
TASK [check test for rc value as integer] *************************************************************************
ok: [foo-1] => {
"bar_status.rc == 127": true
}
常用的事实
以下 Ansible 事实经常在条件语句中使用。
ansible_facts[‘distribution’]
可能的值(示例,并非完整列表)
Alpine
Altlinux
Amazon
Archlinux
ClearLinux
Coreos
CentOS
Debian
Fedora
Gentoo
Mandriva
NA
OpenWrt
OracleLinux
RedHat
Slackware
SLES
SMGL
SUSE
Ubuntu
VMwareESX
ansible_facts[‘distribution_major_version’]
操作系统的主要版本。例如,Ubuntu 16.04 的值为 16。
ansible_facts[‘os_family’]
可能的值(示例,并非完整列表)
AIX
Alpine
Altlinux
Archlinux
Darwin
Debian
FreeBSD
Gentoo
HP-UX
Mandrake
RedHat
SMGL
Slackware
Solaris
Suse
Windows