调试任务

Ansible 提供了一个任务调试器,以便您可以在执行期间修复错误,而无需编辑您的 playbook 并再次运行以查看您的更改是否生效。您可以在任务的上下文中访问调试器的所有功能。您可以检查或设置变量的值,更新模块参数,并使用新的变量和参数重新运行该任务。调试器允许您解决失败的原因并继续执行 playbook。

启用调试器

调试器默认情况下未启用。如果您想在 playbook 执行期间调用调试器,您必须首先启用它。

使用以下三种方法之一启用调试器

  • 使用 debugger 关键字

  • 在配置或环境变量中,或

  • 作为策略

使用 debugger 关键字启用调试器

2.5 版本新增功能。

您可以使用 debugger 关键字来启用(或禁用)特定 play、角色、block 或任务的调试器。当开发或扩展 playbook、play 和角色时,此选项特别有用。您可以在新的或更新的任务上启用调试器。如果它们失败,您可以有效地修复错误。debugger 关键字接受五个值

结果

always

始终调用调试器,无论结果如何

never

永远不要调用调试器,无论结果如何

on_failed

仅在任务失败时调用调试器

on_unreachable

仅当主机无法访问时才调用调试器

on_skipped

仅在跳过任务时调用调试器

当您使用 debugger 关键字时,您指定的值会覆盖任何启用或禁用调试器的全局配置。如果您在多个级别定义 debugger,例如在角色和任务中,Ansible 会遵守最精细的定义。在 play 或角色级别的定义适用于该 play 或角色内的所有 block 和任务,除非它们指定不同的值。在 block 级别的定义会覆盖在 play 或角色级别的定义,并适用于该 block 内的所有任务,除非它们指定不同的值。在任务级别的定义始终适用于该任务;它会覆盖在 block、play 或角色级别的定义。

使用 debugger 关键字的示例

在任务上设置 debugger 关键字的示例

- name: Execute a command
  ansible.builtin.command: "false"
  debugger: on_failed

在 play 上设置 debugger 关键字的示例

- name: My play
  hosts: all
  debugger: on_skipped
  tasks:
    - name: Execute a command
      ansible.builtin.command: "true"
      when: False

在多个级别设置 debugger 关键字的示例

- name: Play
  hosts: all
  debugger: never
  tasks:
    - name: Execute a command
      ansible.builtin.command: "false"
      debugger: on_failed

在此示例中,调试器在 play 级别设置为 never,在任务级别设置为 on_failed。如果任务失败,Ansible 会调用调试器,因为任务的定义会覆盖其父 play 的定义。

在配置或环境变量中启用调试器

2.5 版本新增功能。

您可以使用 ansible.cfg 中的设置或环境变量全局启用任务调试器。唯一的选项是 TrueFalse。如果您将配置选项或环境变量设置为 True,则 Ansible 默认会在失败的任务上运行调试器。

要从 ansible.cfg 启用任务调试器,请将此设置添加到 [defaults] 部分

[defaults]
enable_task_debugger = True

要使用环境变量启用任务调试器,请在运行 playbook 时传递该变量

ANSIBLE_ENABLE_TASK_DEBUGGER=True ansible-playbook -i hosts site.yml

当您全局启用调试器时,每个失败的任务都会调用调试器,除非角色、play、block 或任务显式禁用调试器。如果您需要更精细地控制触发调试器的条件,请使用 debugger 关键字。

将调试器作为策略启用

如果您正在运行旧的 playbook 或角色,您可能会看到调试器作为策略启用。您可以在 play 级别、ansible.cfg 中或使用环境变量 ANSIBLE_STRATEGY=debug 来执行此操作。例如

- hosts: test
  strategy: debug
  tasks:
  ...

或在 ansible.cfg 中

[defaults]
strategy = debug

注意

此向后兼容的方法与 2.5 之前的 Ansible 版本匹配,可能会在未来的版本中删除。

在调试器中解决错误

在 Ansible 调用调试器后,您可以使用七个 调试器命令 来解决 Ansible 遇到的错误。考虑以下示例 playbook,它定义了 var1 变量,但错误地在任务中使用了未定义的 wrong_var 变量。

- hosts: test
  debugger: on_failed
  gather_facts: false
  vars:
    var1: value1
  tasks:
    - name: Use a wrong variable
      ansible.builtin.ping: data={{ wrong_var }}

如果您运行此 playbook,Ansible 会在任务失败时调用调试器。从调试提示符中,您可以更改模块参数或变量并再次运行该任务。

PLAY ***************************************************************************

TASK [wrong variable] **********************************************************
fatal: [192.0.2.10]: FAILED! => {"failed": true, "msg": "ERROR! 'wrong_var' is undefined"}
Debugger invoked
[192.0.2.10] TASK: wrong variable (debug)> p result._result
{'failed': True,
 'msg': 'The task includes an option with an undefined variable. The error '
        "was: 'wrong_var' is undefined\n"
        '\n'
        'The error appears to have been in '
        "'playbooks/debugger.yml': line 7, "
        'column 7, but may\n'
        'be elsewhere in the file depending on the exact syntax problem.\n'
        '\n'
        'The offending line appears to be:\n'
        '\n'
        '  tasks:\n'
        '    - name: wrong variable\n'
        '      ^ here\n'}
[192.0.2.10] TASK: wrong variable (debug)> p task.args
{u'data': u'{{ wrong_var }}'}
[192.0.2.10] TASK: wrong variable (debug)> task.args['data'] = '{{ var1 }}'
[192.0.2.10] TASK: wrong variable (debug)> p task.args
{u'data': '{{ var1 }}'}
[192.0.2.10] TASK: wrong variable (debug)> redo
ok: [192.0.2.10]

PLAY RECAP *********************************************************************
192.0.2.10               : ok=1    changed=0    unreachable=0    failed=0

在调试器中更改任务参数以使用 var1 而不是 wrong_var 使任务成功运行。

可用的调试命令

您可以在调试提示符下使用以下七个命令

命令

快捷方式

操作

print

p

打印有关任务的信息

task.args[key] = value

无快捷方式

更新模块参数

task_vars[key] = value

无快捷方式

更新任务变量(您必须接下来使用 update_task

update_task

u

使用更新的任务变量重新创建任务

redo

r

再次运行该任务

continue

c

继续执行,从下一个任务开始

quit

q

退出调试器

有关更多详细信息,请参阅下面的各个描述和示例。

Update args 命令

task.args[*key*] = *value* 更新模块参数。此示例 playbook 具有无效的软件包名称。

- hosts: test
  strategy: debug
  gather_facts: true
  vars:
    pkg_name: not_exist
  tasks:
    - name: Install a package
      ansible.builtin.apt: name={{ pkg_name }}

当您运行 playbook 时,无效的软件包名称会触发错误,并且 Ansible 会调用调试器。您可以通过查看,然后更新模块参数来修复软件包名称。

[192.0.2.10] TASK: install package (debug)> p task.args
{u'name': u'{{ pkg_name }}'}
[192.0.2.10] TASK: install package (debug)> task.args['name'] = 'bash'
[192.0.2.10] TASK: install package (debug)> p task.args
{u'name': 'bash'}
[192.0.2.10] TASK: install package (debug)> redo

更新模块参数后,使用 redo 再次使用新参数运行该任务。

Update vars 命令

task_vars[*key*] = *value* 更新 task_vars。你可以通过查看并更新任务变量,而不是模块参数来修复上面的 playbook。

[192.0.2.10] TASK: install package (debug)> p task_vars['pkg_name']
u'not_exist'
[192.0.2.10] TASK: install package (debug)> task_vars['pkg_name'] = 'bash'
[192.0.2.10] TASK: install package (debug)> p task_vars['pkg_name']
'bash'
[192.0.2.10] TASK: install package (debug)> update_task
[192.0.2.10] TASK: install package (debug)> redo

在更新任务变量之后,你必须使用 update_task 来加载新的变量,然后才能使用 redo 再次运行任务。

注意

在 2.5 版本中,为了避免与 vars() Python 函数冲突,这里从 vars 更新为了 task_vars

更新任务命令

2.8 版本新增。

uupdate_task 会根据原始任务数据结构和模板,使用更新后的任务变量重新创建任务。有关使用示例,请参阅 更新变量命令

重做命令

rredo 再次运行该任务。

继续命令

ccontinue 继续执行,从下一个任务开始。

退出命令

qquit 退出调试器。playbook 执行将被中止。

调试器如何与 free 策略交互

在启用默认的 linear 策略时,Ansible 会在调试器处于活动状态时暂停执行,并在你输入 redo 命令后立即运行调试过的任务。但是,如果启用 free 策略,Ansible 不会等待所有主机,并且可能会在一个主机上的任务失败之前,在另一个主机上排队后续任务。使用 free 策略时,Ansible 不会在调试器处于活动状态时排队或执行任何任务。但是,所有已排队的任务仍然在队列中,并在你退出调试器后立即运行。如果你使用 redo 从调试器重新调度任务,则其他已排队的任务可能会在你重新调度的任务之前执行。有关策略的更多信息,请参阅 控制 playbook 执行:策略等等

另请参阅

执行用于故障排除的 Playbook

在调试或测试时运行 playbook

Ansible Playbook

playbook 简介

沟通

有疑问?需要帮助?想要分享你的想法?请访问 Ansible 通信指南