指令 host 定义playbook的执行主机范围,与命令模式下的ansible匹配规则一样。

remote_user 定义playbook的执行用户,执行任务也可以定义在任务级别,如: tasks: - name: test connection ping: remote_user: yourname sudo: yes 注意:也可以用sudo指令来说明所有或者部分任务以sudo方式执行

vars 定义变量

vars_files 定义变量文件

tasks 功能 定义任务列表,由模块来执行完成

name 定义playbook或者task的名称

notify 任务执行结果如果是发生更改了的则触发定义在handler的任务执行

handlers 定义被触发的任务列表,由模块来执行完成

include 能包含的包括task,handler和playbook 可以在include的时候传递变量

   示例如下:

tasks: 
  - include: wordpress.yml 
    vars: 
         remote_user: timmy 
         some_list_variable: 
           - alpha 
           - beta 
           - gamma

roles 定义主机对应的角色,角色是一组按照目录组合的配置,ansible自动完成文件搜索,去找对应目录下的main.yml文件来执行,具体目录结构如下:

  • defaults/ 默认的变量保存在这个目录下
  • files/ 文件
  • templates/ 模板
  • tasks/ 任务
  • handlers/ 处理器
  • vars/ 变量
  • meta/ 角色本身的信息,如通过dependencies指令指定依赖
  • library/ 私有模块

另外,你也可以给role传递变量。

这里重点说明下,role是使ansible的状态管理可复用很重要的一个概念,很多时候你只需要在自己的playbook里引用下别人的role即可,大家写的role可以相互共享,相互参考。官方也提供了ansible-galaxy这个命令用于安装社区分享的role,具体可参考Galaxy官网

pre_taskpost_task 在role被应用之前和之后执行的任务列表

变量 变量本来是个很简单的东西,但是变量在Playbook里是个很复杂的,其复杂的原因在于你不知道它从哪里来到哪里去了,下面我们也从这两个方面来剖析它。

从哪里来

inventory

在group和host中都可以定义变量,示例如下:

[atlanta] 
host1 http_port=80 maxRequestsPerChild=808 
host2 http_port=303 maxRequestsPerChild=909 
[southeast:vars] 
some_server=foo.southeast.example.com 
halon_system_timeout=30 
self_destruct_countdown=60 
escape_pods=2

varsvars_files 在playbook的开头可以用这俩指令来定义一些初始变量,这个可以参考上文中的playbook的demo include 通过在playbook的include指令可以其他task的时候,可以给文件传递变量,示例如下:

tasks: 
   - include: wordpress.yml user=timmy

roles 当给一个主机应用角色的时候可以传递变量,然后在角色内使用这些变量,示例如下:

- hosts: webservers 
  roles: 
       - common 
       - { role: foo_app_instance, dir: '/opt/a', port: 5000 }

facts 默认在每次执行playbook前都获取设备信息,所有这些信息都可以作为变量应用到playbook中,要查看这些变量可以执行: ansible cloud -m setup

register 把任务的输出定义为变量,然后用于其他任务,示例如下:

tasks: 
    - shell: /usr/bin/foo 
      register: foo_result 
      ignore_errors: True

内置变量 ansible内置了一些变量以方便主机之间相互调用各自的变量。这些变量包括: * hostvars允许你访问另一个主机的变量,当然前提是ansible已经收集到这个主机的变量了:

  • group_names是当前主机所在的group列表
  • groups是所有inventory的group列表
  • inventory_hostname是在inventory里定义的主机名
  • play_hosts是当前的playbook范围内的主机列表
  • inventory_dir和inventory_file是定义inventory的目录和文件

命令行 在运行playbook的时候也可以传递一些变量供playbook使用,示例如下:

ansible-playbook release.yml --extra-vars "hosts=vipers user=starbuck"

也可以从json文件里读取变量,示例如下;

ansible-playbook main.yml -e "@vars.json"

用到哪里 所有变量都可以在playbook或者jinja2模板中通过{{ varname }}中使用。另外,当变量和jinja2的管道配合起来的时候能提供各种灵活的条件判断和变量处理。具体看下边两个例子。 如果第一个任务执行失败了才会执行第二个任务,可以这么写:

tasks: 
     - shell: /usr/bin/foo 
       register: result 
       ignore_errors: True 
     - debug: msg="it failed" 
       when: result|failed

去重一个列表,可以这么写: {{ list | uniq }} 

变量的优先级

  • -e 命令行指定的最高
  • inventory文件定义的变量次之,其实inventory文件也分全局,group级别的和hosts级别的变量定义
  • fact变量次之
  • 角色的default变量优先级最低

条件

when 可用于task,role和include,在满足条件时task才会被执行。至于when指令后跟的逻辑表达式也是标准的逻辑表达式,示例如下:

tasks: 
     - shell: echo "only on Red Hat 6, derivatives, and later" 
       when: ansible_os_family == "RedHat" and ansible_lsb.major_release|int >= 6 
     - shell: echo "This certainly is epic!" 
       when: epic is defined

循环

标准遍历 用with_items可以遍历一个列表,注意这里只会遍历一层。示例如下:

- name: add several users 
  user: name={{ item }} state=present groups=wheel 
  with_items: 
     - testuser1 
     - testuser2

嵌套遍历

用with_nested可以遍历一个列表,注意这里会遍历多层,直到最内层。示例如下:

- name: give users access to multiple databases 
  mysql_user: name={{ item[0] }} priv={{ item[1] }}.*:ALL append_privs=yes password=foo 
  with_nested: 
     - [ 'alice', 'bob', 'eve' ] 
     - [ 'clientdb', 'employeedb', 'providerdb' ]

遍历字典 用with_dict可以遍历一个字典,用key和value来表示。示例如下: 变量文件

--- 
users: 
  alice: 
     name: Alice Appleworth 
     telephone: 123-456-7890 
  bob: 
     name: Bob Bananarama 
     telephone: 987-654-3210

playbook文件

tasks: 
  - name: Print phone records
    debug: msg="User {{ item.key }} is {{ item.value.name }} ({{ item.value.telephone }})" 
    with_dict: users

文件通配符循环

 用with_fileglob可以获取本地文件列表。示例如下:

# copy each file over that matches the given pattern 
- copy: src={{ item }} dest=/etc/fooapp/ owner=root mode=600 
  with_fileglob: 
      - /playbooks/files/fooapp/*

对齐的列表 用with_together可以达到类似python里的zip函数的功能。示例如下: 变量文件:

--- 
alpha: [ 'a', 'b', 'c', 'd' ] 
numbers: [ 1, 2, 3, 4 ]

playbook文件

tasks: 
  - debug: msg="{{ item.0 }} and {{ item.1 }}" 
    with_together: 
       - alpha 
       - numbers

子元素循环 with_subelements这个比较费解。

数字序列循环 可以通过with_sequence来生成一个数字序列,其参数包括:

  • start起始数字
  • end结束数字
  • stride步长
  • count个数
  • format输出的字符串

示例如下:

--- 
- hosts: all 
  tasks: 
     # create groups 
      - group: name=evens state=present 
      - group: name=odds state=present 
     # create some test users 
      - user: name={{ item }} state=present groups=evens 
       with_sequence: start=0 end=32 format=testuser%02x 
     # create a series of directories with even numbers for some reason 
      - file: dest=/var/stuff/{{ item }} state=directory 
        with_sequence: start=4 end=16 stride=2 
     # a simpler way to use the sequence plugin 
     # create 4 groups 
       - group: name=group{{ item }} state=present 
         with_sequence: count=4

随机循环 通过with_random_choice从一个序列里随机取一个元素。示例如下:

- debug: msg={{ item }} 
  with_random_choice: 
      - "go through the door" 
      - "drink from the goblet" 
      - "press the red button" 
      - "do nothing"

until循环 这种循环由三个指令完成:

  • * until是一个条件表达式,如果满足条件循环结束
  • * retry是重试的次数
  • * delay是延迟时间

示例如下:

- action: shell /usr/bin/foo 
  register: result 
  until: result.stdout.find("all systems go") != -1 
  retries: 5 
  delay: 10

循环直到找到文件 与with_items类似,只是with_first_found找到列表里的第一个文件就会终止循环。示例如下:

- name: INTERFACES | Create Ansible header for /etc/network/interfaces 
  template: src={{ item }} dest=/etc/foo.conf 
  with_first_found: 
      - "{{ansible_virtualization_type}}_foo.conf" 
      - "default_foo.conf"

循环一个task的输出 with_lines指令后跟一个命令,ansible会遍历命令的输出。示例如下:

name: Example of looping over a command result 
 shell: /usr/bin/frobnicate {{ item }} 
 with_lines: /usr/bin/frobnications_per_host --param {{ inventory_hostname }}

-带索引地循环列表 与with_items类似,with_indexed_items会把列表索引和对应元素放到一个列表里。示例如下:

- name: indexed loop demo 
  debug: msg="at array position {{ item.0 }} there is a value {{ item.1 }}"          
  with_indexed_items: some_list

扁平化循环列表 with_flattened会先拍扁一个列表,然后执行with_items。示例如下:

- name: flattened loop demo 
  yum: name={{ item }} state=installed 
  with_flattened: 
    - ['nc','git', ['nmap', 'vim']]

配合register循环列表 register注册一个变量后,可以配合with_items来遍历变量结果。示例如下:

- shell: echo "{{ item }}" 
  with_items: 
      - one 
      - two 
  register: echo 
- name: Fail if return code is not 0 
  fail: 
    msg: "The command ({{ item.cmd }}) did not have a 0 return code" 
  when: item.rc != 0 
  with_items: echo.results