一.循环

循环迭代任务

1.简单循环


loop:             ##赋值列表
- value1
- value2
- ...

{<!-- -->{item}}          ##迭代变量名称

[admin@ansible .ansible]$ ansible westos -m shell -a 'touch /mnt/westos_file2'
172.25.32.11 | CHANGED | rc=0 &gt;&gt;

172.25.32.12 | CHANGED | rc=0 &gt;&gt;

[admin@ansible .ansible]$ cat westos.yml
- name: create file
  hosts: westos
  tasks:
    - name: file module
      file:
        name: /mnt/{<!-- -->{item}}
        state: absent
     
      loop:
        - westos_file1
        - westos_file2

[admin@ansible .ansible]$ ansible-playbook westos.yml

PLAY [create file] ********************************************************************************************************

TASK [Gathering Facts] ****************************************************************************************************
ok: [172.25.32.12]
ok: [172.25.32.11]

TASK [file module] ********************************************************************************************************
ok: [172.25.32.12] =&gt; (item=westos_file1)
ok: [172.25.32.11] =&gt; (item=westos_file1)
changed: [172.25.32.11] =&gt; (item=westos_file2)
changed: [172.25.32.12] =&gt; (item=westos_file2)

PLAY RECAP ****************************************************************************************************************
172.25.32.11               : ok=2    changed=1    unreachable=0    failed=0    skipped=0    rescued=0    ignored=0  
172.25.32.12               : ok=2    changed=1    unreachable=0    failed=0    skipped=0    rescued=0    ignored=0

2.循环散列或字典列表


[admin@ansible .ansible]$ cat westos1.yml
- name: create file
  hosts: westos
  tasks:
    - name: file module
      service:
        name: "{<!-- -->{ item.name}}"
        state: "{<!-- -->{ item.state }}"
      loop:
        - name: httpd
          state: stopped
        - name: vsftpd
          state: started

[admin@ansible .ansible]$ ansible-playbook westos1.yml

PLAY [create file] ********************************************************************************************************

TASK [Gathering Facts] ****************************************************************************************************
ok: [172.25.32.12]
ok: [172.25.32.11]

TASK [file module] ********************************************************************************************************
changed: [172.25.32.12] =&gt; (item={u'state': u'stopped', u'name': u'httpd'})
changed: [172.25.32.11] =&gt; (item={u'state': u'stopped', u'name': u'httpd'})
changed: [172.25.32.11] =&gt; (item={u'state': u'started', u'name': u'vsftpd'})
changed: [172.25.32.12] =&gt; (item={u'state': u'started', u'name': u'vsftpd'})

PLAY RECAP ****************************************************************************************************************
172.25.32.11               : ok=2    changed=1    unreachable=0    failed=0    skipped=0    rescued=0    ignored=0  
172.25.32.12               : ok=2    changed=1    unreachable=0    failed=0    skipped=0    rescued=0    ignored=0

二.条件


when:
  - 条件1
  - 条件2

条件判断
=                 value == "字符串",value == 数字
&lt;                 value &lt;  数字
&gt;                 value &gt;  数字
&lt;=                value &lt;= 数字
&gt;=                value &gt;= 数字
!=                value != 数字
is defined value   value is defined         变量存在
is not defined     value is not defined     变量不存在
in                value is in value       变量为    
not in            value is not in value   变量不为
bool变量 为true    value                   value的值为true
bool变量 false     not value               value的值为false
                  value in value2         value的值在value2列表中

多条条件组合


when:
  条件1 and 条件2
  - 条件1
  - 条件2

when:
  条件1 or 条件2

when: &gt;
  条件1
  or
  条件2

测试admin用户是否存在
[admin@ansible .ansible]$ cat westos.yml
- name: test
  hosts: westos
  tasks:
    - name: check user
      shell: id admin
      ignore_errors: yes
      register: admin

    - name: debug
      debug:
        msg: admin user is exist
      when:
        - admin.rc == 0

[admin@ansible .ansible]$ ansible-playbook westos.yml

PLAY [test] ***************************************************************************************************************

TASK [Gathering Facts] ****************************************************************************************************
ok: [172.25.32.12]
ok: [172.25.32.11]

TASK [check user] *********************************************************************************************************
changed: [172.25.32.11]
changed: [172.25.32.12]

TASK [debug] **************************************************************************************************************
ok: [172.25.32.11] =&gt; {
    "msg": "admin user is exist"
}
ok: [172.25.32.12] =&gt; {
    "msg": "admin user is exist"
}

PLAY RECAP ****************************************************************************************************************
172.25.32.11               : ok=3    changed=1    unreachable=0    failed=0    skipped=0    rescued=0    ignored=0  
172.25.32.12               : ok=3    changed=1    unreachable=0    failed=0    skipped=0    rescued=0    ignored=0

三.触发器


notify:   触发器当遇到更改是触发handlers
handlers: 触发器触发后执行的动作

注意:触发器没有遇到更改时就不会触发也就不会执行下边的动作,遇到更改时会触发且执行下边的动作

[admin@ansible .ansible]$ cat apache_port.yml
- name: port check
  hosts: westos
  tasks:
    - name: configure httpd.conf
      lineinfile:
        path: /etc/httpd/conf/httpd.conf
        regexp: "^Listen"
        line: "Listen 8080"
      notify: restart httpd

  handlers:
    - name: restart httpd
      service:
        name: httpd
        state: restarted

    - name: restart vsftpd
      service:
        name: vsftpd
        state: restarted

[admin@ansible .ansible]$ ansible-playbook apache_port.yml

PLAY [port check] *********************************************************************************************************

TASK [Gathering Facts] ****************************************************************************************************
ok: [172.25.32.12]
ok: [172.25.32.11]

TASK [configure httpd.conf] ***********************************************************************************************
changed: [172.25.32.12]
changed: [172.25.32.11]

RUNNING HANDLER [restart httpd] *******************************************************************************************
changed: [172.25.32.12]
changed: [172.25.32.11]

PLAY RECAP ****************************************************************************************************************
172.25.32.11               : ok=3    changed=2    unreachable=0    failed=0    skipped=0    rescued=0    ignored=0  
172.25.32.12               : ok=3    changed=2    unreachable=0    failed=0    skipped=0    rescued=0    ignored=0  

被控主机查看端口
[root@www ~]# cat /etc/httpd/conf/httpd.conf
......
#Listen 12.34.56.78:80
Listen 8080
......

四.处理失败任务

1.ignore_errors
作用:
当play遇到任务失败是会终止
ignore_errors: yes     将会忽略任务失败使下面的任务继续运行


[admin@ansible .ansible]$ cat ignore_errors.yml
- name: test
  hosts: westos
  tasks:
    - name: test play
      dnf:
        name: westos
        state: latest
      ignore_errors: yes

    - name: create file
      file:
        path: /mnt/file
        state: touch

[admin@ansible .ansible]$ ansible-playbook ignore_errors.yml

PLAY [test] ***************************************************************************************************************

TASK [Gathering Facts] ****************************************************************************************************
ok: [172.25.32.12]
ok: [172.25.32.11]

TASK [test play] **********************************************************************************************************
fatal: [172.25.32.12]: FAILED! =&gt; {"changed": false, "cmd": "dnf install -y python2-dnf", "msg": "[Errno 2] No such file or directory", "rc": 2}
...ignoring
fatal: [172.25.32.11]: FAILED! =&gt; {"changed": false, "cmd": "dnf install -y python2-dnf", "msg": "[Errno 2] No such file or directory", "rc": 2}
...ignoring

TASK [create file] ********************************************************************************************************
changed: [172.25.32.11]
changed: [172.25.32.12]

PLAY RECAP ****************************************************************************************************************
172.25.32.11               : ok=3    changed=1    unreachable=0    failed=0    skipped=0    rescued=0    ignored=1  
172.25.32.12               : ok=3    changed=1    unreachable=0    failed=0    skipped=0    rescued=0    ignored=1

2.force_handlers

作用:  当任务失败后play被终止也会调用触发器进程


- name: port check
  hosts: westos
  force_handlers: yes
  tasks:
    - name: configure httpd.conf
      lineinfile:
        path: /etc/httpd/conf/httpd.conf
        regexp: "^Listen"
        line: "Listen 8080"
      notify:
        - restart httpd
        - restart vsftpd
    - name: install error
      dnf:
        name: westos
        state: latest
 
  handlers:
    - name: restart httpd
      service:
        name: httpd
        state: restarted
   
    - name: restart vsftpd
      service:
        name: vsftpd
        state: restarted

[admin@ansible .ansible]$ ansible-playbook apache_port.yml

PLAY [port check] *********************************************************************************************************

TASK [Gathering Facts] ****************************************************************************************************
ok: [172.25.32.12]
ok: [172.25.32.11]

TASK [configure httpd.conf] ***********************************************************************************************
changed: [172.25.32.11]
changed: [172.25.32.12]

TASK [install error] ******************************************************************************************************
fatal: [172.25.32.11]: FAILED! =&gt; {"changed": false, "cmd": "dnf install -y python2-dnf", "msg": "[Errno 2] No such file or directory", "rc": 2}
fatal: [172.25.32.12]: FAILED! =&gt; {"changed": false, "cmd": "dnf install -y python2-dnf", "msg": "[Errno 2] No such file or directory", "rc": 2}

RUNNING HANDLER [restart httpd] *******************************************************************************************
changed: [172.25.32.12]
changed: [172.25.32.11]

RUNNING HANDLER [restart vsftpd] ******************************************************************************************
changed: [172.25.32.11]
changed: [172.25.32.12]

PLAY RECAP ****************************************************************************************************************
172.25.32.11               : ok=4    changed=3    unreachable=0    failed=1    skipped=0    rescued=0    ignored=0  
172.25.32.12               : ok=4    changed=3    unreachable=0    failed=1    skipped=0    rescued=0    ignored=0

3.changed_when

作用:  控制任务在何时报告它已进行更改


[admin@ansible .ansible]$ cat apache_port.yml
- name: port check
  hosts: westos
  force_handlers: yes
  tasks:
    - name: configure httpd.conf
      lineinfile:
        path: /etc/httpd/conf/httpd.conf
        regexp: "^Listen"
        line: "Listen 8080"
      changed_when: yes
      notify:
        - restart httpd
        - restart vsftpd
   
  handlers:
    - name: restart httpd
      service:
        name: httpd
        state: restarted
   
    - name: restart vsftpd
      service:
        name: vsftpd
        state: restarted

不论文件有没有作修改,他都会认为文件已修改,且执行。
[admin@ansible .ansible]$ ansible-playbook apache_port.yml

PLAY [port check] *********************************************************************************************************

TASK [Gathering Facts] ****************************************************************************************************
ok: [172.25.32.12]
ok: [172.25.32.11]

TASK [configure httpd.conf] ***********************************************************************************************
changed: [172.25.32.12]
changed: [172.25.32.11]

RUNNING HANDLER [restart httpd] *******************************************************************************************
changed: [172.25.32.12]
changed: [172.25.32.11]

RUNNING HANDLER [restart vsftpd] ******************************************************************************************
changed: [172.25.32.12]
changed: [172.25.32.11]

PLAY RECAP ****************************************************************************************************************
172.25.32.11               : ok=4    changed=3    unreachable=0    failed=0    skipped=0    rescued=0    ignored=0  
172.25.32.12               : ok=4    changed=3    unreachable=0    failed=0    skipped=0    rescued=0    ignored=0

4.failed_when

当符合条件时强制任务失败


[admin@ansible .ansible]$ cat ignore_errors.yml
- name: test
  hosts: westos
  force_handlers: yes
  tasks:
    - name: test play
      dnf:
        name: westos
        state: latest
      failed_when: no

    - name: create file
      file:
        path: /mnt/file1
        state: touch

不论任务执行成功与否,都视为成功。
[admin@ansible .ansible]$ ansible-playbook ignore_errors.yml

PLAY [test] ***************************************************************************************************************

TASK [Gathering Facts] ****************************************************************************************************
ok: [172.25.32.12]
ok: [172.25.32.11]

TASK [test play] **********************************************************************************************************
ok: [172.25.32.11]
ok: [172.25.32.12]

TASK [create file] ********************************************************************************************************
changed: [172.25.32.12]
changed: [172.25.32.11]

PLAY RECAP ****************************************************************************************************************
172.25.32.11               : ok=3    changed=1    unreachable=0    failed=0    skipped=0    rescued=0    ignored=0  
172.25.32.12               : ok=3    changed=1    unreachable=0    failed=0    skipped=0    rescued=0    ignored=0

5.block


block:    ##定义要运行的任务
rescue:   ##定义当block句子中出现失败任务后运行的任务
always:   ##定义最终独立运行的任务

当任务中block执行没问题,之后执行always。当block有问题则执行rescue,然后执行always。也就是说不论执行的是block还是rescue,之后都会执行always。

[admin@ansible .ansible]$ cat westos.yml
- name: test
  hosts: westos
  tasks:
    - name: test play
      block:
        - name: install httpd
          yum:
            name: westos
            state: latest    
      rescue:
        - name: install vsftpd
          yum:
            name: vsftpd
            state: latest
      always:
        - name: hello westos
          debug:
            msg: hello westos

当block执行失败时,执行rescue,之后执行always
[admin@ansible .ansible]$ ansible-playbook westos.yml

PLAY [test] ***************************************************************************************************************

TASK [Gathering Facts] ****************************************************************************************************
ok: [172.25.32.12]
ok: [172.25.32.11]

TASK [install httpd] ******************************************************************************************************
fatal: [172.25.32.11]: FAILED! =&gt; {"changed": false, "msg": "No package matching 'westos' found available, installed or updated", "rc": 126, "results": ["No package matching 'westos' found available, installed or updated"]}
fatal: [172.25.32.12]: FAILED! =&gt; {"changed": false, "msg": "No package matching 'westos' found available, installed or updated", "rc": 126, "results": ["No package matching 'westos' found available, installed or updated"]}

TASK [install vsftpd] *****************************************************************************************************
ok: [172.25.32.11]
ok: [172.25.32.12]

TASK [hello westos] *******************************************************************************************************
ok: [172.25.32.11] =&gt; {
    "msg": "hello westos"
}
ok: [172.25.32.12] =&gt; {
    "msg": "hello westos"
}

PLAY RECAP ****************************************************************************************************************
172.25.32.11               : ok=3    changed=0    unreachable=0    failed=0    skipped=0    rescued=1    ignored=0  
172.25.32.12               : ok=3    changed=0    unreachable=0    failed=0    skipped=0    rescued=1    ignored=0

[admin@ansible .ansible]$ cat westos.yml
- name: test
  hosts: westos
  tasks:
    - name: test play
      block:
        - name: install httpd
          yum:
            name: westos
            state: latest
          failed_when: no
      rescue:
        - name: install vsftpd
          yum:
            name: vsftpd
            state: latest
      always:
        - name: hello westos
          debug:
            msg: hello westos

当block执行成功时,跳过rescue,然后执行always
[admin@ansible .ansible]$ ansible-playbook westos.yml

PLAY [test] ***************************************************************************************************************

TASK [Gathering Facts] ****************************************************************************************************
ok: [172.25.32.12]
ok: [172.25.32.11]

TASK [install httpd] ******************************************************************************************************
ok: [172.25.32.12]
ok: [172.25.32.11]

TASK [hello westos] *******************************************************************************************************
ok: [172.25.32.11] =&gt; {
    "msg": "hello westos"
}
ok: [172.25.32.12] =&gt; {
    "msg": "hello westos"
}

PLAY RECAP ****************************************************************************************************************
172.25.32.11               : ok=3    changed=0    unreachable=0    failed=0    skipped=0    rescued=0    ignored=0  
172.25.32.12               : ok=3    changed=0    unreachable=0    failed=0    skipped=0    rescued=0    ignored=0

#测试练习#
建立playbook ~/westos.yml要求如下:
建立大小为1500M名为/dev/vdb1的设备
如果/dev/vdb不存在请输入:
/dev/vdb is not exist
如果/dev/vdb大小不足2G请输出:
/dev/vdb is less then 2G
并建立800M大小的/dev/vdb1
此设备挂载到/westos上


[admin@ansible .ansible]$ cat westos.yml
- name: create lvm
  hosts: all
  tasks:
    - name: check /dev/vdb
      debug:
        msg: /dev/vdb is not exist
      when: ansible_facts['devices']['vdb'] is not defined
   
    - name: create /dev/vdb1
      block:
        - name: create 1500M
          parted:
            device: /dev/vdb
            number: 1
            state: present
            part_end: 1500MB
          when: ansible_facts['devices']['vdb'] is defined

      rescue:
        - name: create 800MB
          parted:
            device: /dev/vdb
            number: 1
            state: present
            part_end: 800MB
          when: ansible_facts['devices']['vdb'] is defined

      always:
        - name: mkfs.xfs /dev/vdb1
          filesystem:
            fstype: xfs
            dev: /dev/vdb1
        - name: mkdir /westos
          file:
            path: /westos
            state: directory
        - name: mount /dev/vdb1
          mount:
            path: /westos
            src: /dev/vdb1
            fstype: xfs
            state: mounted
          when: ansible_facts['devices']['vdb'] is defined

[admin@ansible .ansible]$ ansible-playbook westos.yml

PLAY [create lvm] *********************************************************************************************************

TASK [Gathering Facts] ****************************************************************************************************
ok: [172.25.32.12]
ok: [172.25.32.11]

TASK [check /dev/vdb] *****************************************************************************************************
ok: [172.25.32.11] =&gt; {
    "msg": "/dev/vdb is not exist"
}
ok: [172.25.32.12] =&gt; {
    "msg": "/dev/vdb is not exist"
}

TASK [create 1500M] *******************************************************************************************************
skipping: [172.25.32.11]
skipping: [172.25.32.12]

TASK [mkfs.xfs /dev/vdb1] *************************************************************************************************
fatal: [172.25.32.11]: FAILED! =&gt; {"changed": false, "msg": "Device /dev/vdb1 not found."}
fatal: [172.25.32.12]: FAILED! =&gt; {"changed": false, "msg": "Device /dev/vdb1 not found."}

PLAY RECAP ****************************************************************************************************************
172.25.32.11               : ok=2    changed=0    unreachable=0    failed=1    skipped=1    rescued=0    ignored=0  
172.25.32.12               : ok=2    changed=0    unreachable=0    failed=1    skipped=1    rescued=0    ignored=0

云野 » Ansible-任务执行控制

发表回复