变量概述、facts变量、自定义变量、清单变量、剧本变量、防火墙模块、template模块、错误处理、触发执行任务、when条件

playbook 

模块

# 在webservers组中的主机上,安装httpd、php、php-mysqlnd
[root@pubserver ansible]# vim pkg.yml
---
- name: install pkgs
  hosts: webservers
  tasks:
    - name: install web pkgs  # 此任务通过yum安装三个包
      yum:
        name: httpd,php,php-mysqlnd
        state: present
# 安装多个软件包,还可以写为:
---
- name: install pkgs
  hosts: webservers
  tasks:
    - name: install web pkgs
      yum:
        name: [httpd,php,php-mysqlnd]
        state: present
# 安装多个软件包,还可以写为:
---
- name: install pkgs
  hosts: webservers
  tasks:
    - name: install web pkgs
      yum:
        name: 
          - httpd
          - php
          - php-mysqlnd
        state: present
# 根据功能等,可以将一系列软件放到一个组中,安装软件包组,将会把很多软件一起安装上。比如gcc、java等都是开发工具,安装开发工具包组,将会把它们一起安装。
[root@node1 ~]# yum grouplist   # 列出所有的软件包组
[root@node1 ~]# yum groupinstall "Development Tools"
# 如果列出的组名为中文,可以这样进行:
[root@node1 ~]# LANG=C yum grouplist
# 继续编辑pkg.yml,在webservers组中的主机上安装Development tools组
[root@pubserver ansible]# vim pkg.yml
---
- name: install pkgs
  hosts: webservers
  tasks:
    - name: install web pkgs    # 此任务通过yum安装三个包
      yum:
        name:
          - httpd
          - php
          - php-mysqlnd
        state: present
    - name: install dev group    # 此任务通过yum安装一组包
      yum:
        name: "@Development Tools"   # @表示后面的名字是组名
        state: present
[root@pubserver ansible]# ansible-playbook pkg.yml
# 系统升级命令
[root@pubserver ansible]# yum update
# 继续编辑pkg.yml,在webservers组中的主机上升级所有的包到最新版本
---
- name: install pkgs
  hosts: webservers
  tasks:
    - name: install web pkgs
      yum:
        name:
          - httpd
          - php
          - php-mysqlnd
        state: present
    - name: install dev group
      yum:
        name: "@Development Tools"
        state: present
    - name: update system    # 相当于yum update命令
      yum:
        name: "*"       # 表示系统已经安装的所有包
        state: latest
[root@pubserver ansible]# ansible-playbook pkg.yml

ansible变量

facts变量

  • facts翻译过来就是事实。
  • facts变量是ansible自带的预定义变量,用于描述被控端软硬件信息。
  • facts变量通过setup模块获得。
# 通过setup模块查看所有facts变量

[root@pubserver ansible]# ansible webservers -m setup
  • facts变量是一个大的由{}构成的键值对字典。在{}中,有很多层级的嵌套。可以通过参数过滤出第一个层级的内容。
# 查看所有的IPV4地址,filter是过滤的意思
[root@pubserver ansible]# ansible webservers -m setup -a "filter=ansible_all_ipv4_addresses"


# 查看可用内存
[root@pubserver ansible]# ansible webservers -m setup -a "filter=ansible_memfree_mb"
  • 常用的facts变量

    • ansible_all_ipv4_addresses:所有的IPV4地址
    • ansible_bios_version:BIOS版本信息
    • ansible_memtotal_mb:总内存大小
    • ansible_hostname:主机名
  • 在playbook中使用变量

# 显示远程主机的主机名和内存大小。在ansible中,变量使用{{}}表示

# debug模块用于输出信息,常用的参数是msg,用于输出指定内容

[root@pubserver ansible]# vim debug.yml

---

- name: display host info

  hosts: webservers

  tasks:

    - name: display hostname and memory

      debug:    # debug是模块,它的选项msg可以输出指定信息

        msg: "hostname: {{ansible_hostname}}; mem: {{ansible_memtotal_mb}} MB"


[root@pubserver ansible]# ansible-playbook debug.yml

自定义变量

  • 引入变量,可以方便Playbook重用。比如装包的playbook,包名使用变量。多次执行playbook,只要改变变量名即可,不用编写新的playbook。

  • ansible支持10种以上的变量定义方式。常用的变量来源如下:

    • inventory变量。变量来自于主机清单文件
    • facts变量。
    • playbook变量。变量在playbook中定义。
    • 变量文件。专门创建用于保存变量的文件。推荐变量写入单独的文件。
# 使用inventory变量。

[root@pubserver ansible]# vim inventory

[webservers]

web1

web2


[dbs]

db1 username="wangwu"   # 定义主机变量的方法


[webservers:vars]       # 定义组变量的方法,:vars是固定格式

username="zhaoliu"



# 通过变量创建用户

[root@pubserver ansible]# vim var1.yml

---

- name: webservers create user

  hosts: webservers

  tasks:

    - name: create user

      user:

        name: "{{username}}"

        state: present

        

- name: create user in dbs

  hosts: dbs

  tasks:

    - name: create some users

      user:

        name: "{{username}}"

        state: present


[root@pubserver ansible]# ansible-playbook var1.yml


# 上述两个play也可以合并为一个,如下:

[root@pubserver ansible]# vim var1.yml

---

- name: webservers create user

  hosts: dbs,webservers    # 指定执行的目标是webservers组和webservers组

  tasks:

    - name: create user

      user:

        name: "{{username}}"

        state: present

        


# 在playbook中定义变量

# 在webservers组中的主机上创建用户jack,他的密码是123456

[root@pubserver ansible]# vim user_jack.yml

---

- name: create user

  hosts: webservers

  vars:    # 固定格式,用于声明变量

    username: "jack"    # 此处引号可有可无

    mima: "123456"      # 此处引号是需要的,表示数字字符

  tasks:

    - name: create some users

      user:

        name: "{{username}}"   # {}出现在开头,必须有引号

        state: present

        password: "{{mima|password_hash('sha512')}}"


[root@pubserver ansible]# ansible-playbook user_jack.yml



# 将变量定义在文件中

[root@pubserver ansible]# vim vars.yml   # 文件名自定义

---

yonghu: rose

mima: abcd

[root@pubserver ansible]# vim user_rose.yml 

---

- name: create user

  hosts: webservers

  vars_files: vars.yml   # vars_files用于声明变量文件

  tasks:

    - name: create some users

      user:

        name: "{{yonghu}}"   # 这里的变量来自于vars.yml

        state: present

        password: "{{mima|password_hash('sha512')}}"


[root@pubserver ansible]# ansible-playbook user_rose.yml 

补充模块

firewalld模块
  • 用于配置防火墙的模块

  • 常用选项:

    • port:声明端口
    • permanent:永久生效,但不会立即生效
    • immediate:立即生效,临时生效
    • state:enabled,放行;disabled拒绝
  • 防火墙一般默认拒绝,明确写入允许的服务。

  • 有一些服务有名字,有些服务没有名字。但是最终都是基于TCP或UDP的某些端口。比如http服务基于TCP80端口。服务名和端口号对应关系的说明文件是:/etc/services

  • 配置服务器的防火墙,一般来说只要配置开放哪些服务或端口即可。没有明确开放的,都默认拒绝。

  • 应用

    • 在webservers组中的主机上安装并启动nginx
    • 客户端访问服务器的nginx服务
    • 在webservers组中的主机上安装并启动firewalld
    • 客户端访问服务器的nginx服务
    • 在webservers组中的主机上开放nginx服务
# 配置nginx服务

[root@pubserver ansible]# vim firewall.yml

---

- name: configure webservers

  hosts: webservers

  tasks:

    - name: install nginx pkg   # 这里通过yum模块装httpd

      yum:

        name: nginx

        state: present


    - name: start nginx service   # 这里通过service模块启httpd服务

      service:

        name: nginx

        state: started

        enabled: yes

        

[root@pubserver ansible]# ansible-playbook firewall.yml

[root@pubserver ansible]# curl http://192.168.88.11/  # 可访问


# 安装并启动firewalld

[root@pubserver ansible]# vim firewall.yml

---

- name: configure webservers

  hosts: webservers

  tasks:

    - name: install nginx pkg   # 这里通过yum模块装httpd

      yum:

        name: nginx

        state: present


    - name: start nginx service   # 这里通过service模块启httpd服务

      service:

        name: nginx

        state: started

        enabled: yes

  

    - name: install firewalld pkg   # 这里通过yum模块装firewalld

      yum:

        name: firewalld

        state: present


    - name: start firewalld service   # 这里通过service模块启firewalld服务

      service:

        name: firewalld

        state: started

        enabled: yes

  

[root@pubserver ansible]# ansible-playbook firewall.yml

[root@pubserver ansible]# curl http://192.168.88.11/  # 被拒绝

curl: (7) Failed to connect to 192.168.88.11 port 80: 没有到主机的路由


# 配置防火墙规则,放行http协议

[root@pubserver ansible]# vim firewall.yml

---

- name: configure webservers

  hosts: webservers

  tasks:

    - name: install nginx pkg   # 这里通过yum模块装httpd

      yum:

        name: nginx

        state: present


    - name: start nginx service   # 这里通过service模块启httpd服务

      service:

        name: nginx

        state: started

        enabled: yes

  

    - name: install firewalld pkg   # 这里通过yum模块安装firewalld

      yum:

        name: firewalld

        state: present


    - name: start firewalld service   # 这里通过service模块启service服务

      service:

        name: firewalld

        state: started

        enabled: yes

  

    - name: set firewalld rules   # 通过firewalld模块开放80端口

      firewalld:

        port: 80/tcp

        permanent: yes

        immediate: yes

        state: enabled


[root@pubserver ansible]# ansible-playbook firewall.yml 

[root@pubserver ansible]# curl http://192.168.88.11/  # 可访问
template模块
  • copy模块可以上传文件,但是文件内容固定

  • template模块可以上传具有特定格式的文件(如文件中包含变量)

  • 当远程主机接收到文件之后,文件中的变量将会变成具体的值

  • template模块上传的文件,使用的语法叫Jinja2。

  • 常用选项:

    • src:要上传的文件
    • dest:目标文件路径
# 使用template模块将含有变量的文件上传到webservers组中的主机

[root@pubserver ansible]# vim index.html

Welcome to {{ansible_hostname}} on {{ansible_eth0.ipv4.address}}


[root@pubserver ansible]# vim templ.yml

---

- name: upload index

  hosts: webservers

  tasks:

    - name: create web index

      template:

        src: index.html

        dest: /usr/share/nginx/html/index.html


[root@pubserver ansible]# ansible-playbook templ.yml

[root@pubserver ansible]# curl http://192.168.88.11/

Welcome to web1 on 192.168.88.11

[root@pubserver ansible]# curl http://192.168.88.12

Welcome to web2 on 192.168.88.12

[root@web1 ~]# cat /usr/share/nginx/html/index.html

Welcome to web1 on 192.168.88.11

[root@web2 ~]# cat /usr/share/nginx/html/index.html

Welcome to web2 on 192.168.88.12

进阶语法

错误处理

  • 当Playbook中包含很多任务时,当某一个任务遇到错误,它将崩溃,终止执行
# 在webservers组中的主机上启动mysqld服务,然后创建/tmp/service.txt

# 因为目标主机上没有mysqld服务,所以它将崩溃,终止执行。即,不会创建/tmp/service.txt文件

[root@pubserver ansible]# vim myerr.yml

---

- name: my errors

  hosts: webservers

  tasks:

    - name: start mysqld service  # 通过service模块启动mysqld服务

      service:

        name: mysqld

        state: started

        enabled: yes

        

    - name: touch a file   # 通过file模块创建文件

      file:

        path: /tmp/service.txt

        state: touch


# 执行playbook,第1个任务就会失败

[root@pubserver ansible]# ansible-playbook myerr.yml

# 到node1上查看,因为第2个任务没有执行,所以文件不会创建

[root@web1 ~]# ls /tmp/service.txt

ls: cannot access '/tmp/service.txt': No such file or directory
  • 可以指定某一个任务如果出现错误,则忽略它
# 编辑myerr.yml,如果myslqd服务无法启动,则忽略它

[root@pubserver ansible]# vim myerr.yml

---

- name: my errors

  hosts: webservers

  tasks:

    - name: start mysqld service

      service:

        name: mysqld

        state: started

        enabled: yes

      ignore_errors: yes    # 即使这个任务失败了,也要继续执行下去


    - name: touch a file

      file:

        path: /tmp/service.txt

        state: touch


[root@pubserver ansible]# ansible-playbook myerr.yml

[root@web1 ~]# ls /tmp/service.txt   # 第2个任务已执行

/tmp/service.txt
  • 通过全局设置,无论哪个任务出现问题,都要忽略
[root@pubserver ansible]# vim myerr.yml

---

- name: my errors

  hosts: webservers

  ignore_errors: yes

  tasks:

    - name: start mysqld service

      service:

        name: mysqld

        state: started

        enabled: yes


    - name: touch a file

      file:

        path: /tmp/mysql.txt

        state: touch


[root@pubserver ansible]# ansible-playbook myerr.yml

[root@web1 ~]# ls /tmp/mysql.txt 

/tmp/mysql.txt

触发执行任务

  • 通过handlers定义触发执行的任务
  • handlers中定义的任务,不是一定会执行的
  • 在tasks中定义的任务,通过notify关键通知handlers中的哪个任务要执行
  • 只有tasks中的任务状态是changed才会进行通知。
# 下载web1上的/etc/nginx/nginx.conf

[root@pubserver ansible]# vim get_conf.yml

---

- name: download nginx.conf

  hosts: web1

  tasks:

    - name: get nginx.conf

      fetch:

        src: /etc/nginx/nginx.conf

        dest: ./

        flat: yes    # 直接下载文件,不要目录

[root@pubserver ansible]# ansible-playbook get_conf.yml


# 修改nginx.conf的端口为变量

[root@pubserver ansible]# vim +39 nginx.conf

... ...

    server {

        listen       {{http_port}} default_server;

        listen       [::]:{{http_port}} default_server;

        server_name  _;

... ...


# 修改nginx服务的端口为8000,重启nginx

[root@pubserver ansible]# vim trigger.yml

---

- name: configure nginx

  hosts: webservers

  vars:

    http_port: "8000"   # 定义nginx.conf中的变量和值

  tasks:

    - name: upload nginx.conf   # 上传nginx.conf

      template:

        src: ./nginx.conf

        dest: /etc/nginx/nginx.conf


    - name: restart nginx    # 重启服务

      service:

        name: nginx

        state: restarted

# 第一次执行trigger.yml,上传文件和重启服务两个任务的状态都是黄色changed

[root@pubserver ansible]# ansible-playbook trigger.yml

# 第二次执行trigger.yml,上传文件的任务状态是绿色的ok,重启服务任务的状态是黄色changed

[root@pubserver ansible]# ansible-playbook trigger.yml


# 既然配置文件没有改变,那么服务就不应该重启

# 修改Playbook,只有配置文件变化了,才重启服务

[root@pubserver ansible]# vim trigger.yml

---

- name: configure nginx

  hosts: webservers

  vars:

    http_port: "80"

  tasks:

    - name: upload nginx.conf

      template:

        src: ./nginx.conf

        dest: /etc/nginx/nginx.conf

      notify: restart nginx   # 通知restart httpd需要执行


  handlers:

    - name: restart nginx

      service:

        name: nginx

        state: restarted

# 第一次运行Playbook,因为第1个任务是黄色的changed,所以handlers中的任务也被触发执行

[root@pubserver ansible]# ansible-playbook trigger.yml 

# 第二次运行Playbook,因为第1个任务是绿色的OK,也就不会再触发执行其他任务了

[root@pubserver ansible]# ansible-playbook trigger.yml 

when条件

  • 只有满足某一条件时,才执行任务

  • 常用的操作符:

    • ==:相等
    • !=:不等
    • >:大于
    • <:小于
    • <=:小于等于
    • >=:大于等于
  • 多个条件或以使用and或or进行连接

  • when表达式中的变量,可以不使用{{}}

# 当dbs组中的主机内存大于2G的时候,才安装mysql-server

[root@pubserver ansible]# vim when1.yml

---

- name: install mysql-server

  hosts: dbs

  tasks:

    - name: install mysql-server pkg

      yum:

        name: mysql-server

        state: present

      when: ansible_memtotal_mb>2048


# 如果目标主机没有2GB内存,则不会安装mysqld-server

[root@pubserver ansible]# ansible-playbook when1.yml




# 多条件。系统发行版是Rocky8才执行任务

# /etc/motd中的内容,将会在用户登陆时显示在屏幕上

[root@pubserver ansible]# vim motd

 _____________

< hello world >

 -------------

        \   ^__^

         \  (oo)\_______

            (__)\       )\/\

                ||----w |

                ||     ||


[root@pubserver ansible]# vim when2.yml

---

- name: when condition

  hosts: webservers

  tasks:

    - name: modify /etc/motd

      copy:

        dest: /etc/motd

        src: motd

      when: >     # 以下三行合并成一行

        ansible_distribution == "Rocky"

        and

        ansible_distribution_major_version == "8"


[root@pubserver ansible]# ansible-playbook when2.yml

regitster注册变量

Ansible的“register”模块是用于捕获和保存任务执行结果的,它允许将其他任务的输出作为变量使用。register是一个关键字,可以将任务执行的结果赋值给指定的变量名称。这个变量可以在后续任务中使用。 register模块可以捕获各种类型的输出,包括stdout、stderr、rc、changed等。它可以与其他模块一起使用,例如“when”条件、“loop”循环等。

# 在web1组的主机上执行任务,创建/tmp/regfile1.txt,并打印创建结果

[root@pubserver ansible]# vim reg1.yml

---

- name: create file /tmp/regfile1.txt

  hosts: web1

  tasks:

    - name: create file

      file:

        path: /tmp/rgefile1.txt

        state: touch

      register: result


    - name: display output

      debug:

        msg: "{{result}}"


# 在web1主机上执行任务,创建文件/tmp/ademo/abc。如果创建不成功,则通过debug输出create failed

[root@pubserver ansible]# vim reg2.yml

---

- name: create file /tmp/ademo/abc

  hosts: web1

  ignore_errors: yes

  tasks:

    - name: create file

      file:

        path: /tmp/ademo/abc

        state: touch

      register: result


    - name: debug output

      debug:

        msg: "create failed"

      when: result.failed

 cowsay:

# cowsay是一个软件,可以在以下地址获得

# https://rpmfind.net/linux/epel/8/Everything/x86_64/Packages/c/cowsay-3.04-16.el8.noarch.rpm

# 使用方式如下:

[root@pubserver ansible]# yum install -y /root/cowsay-3.04-16.el8.noarch.rpm

[root@pubserver ansible]# cowsay ni hao a

__________

< ni hao a >

----------

  \   ^__^

   \  (oo)\_______

      (__)\       )\/\

          ||----w |

          ||     ||


# 查看其他形象

[root@pubserver ansible]# cowsay -l

Cow files in /usr/share/cowsay:

beavis.zen blowfish bud-frogs bunny cheese cower default dragon

dragon-and-cow elephant elephant-in-snake eyes flaming-sheep fox

ghostbusters head-in hellokitty kiss kitty koala kosh luke-koala

mech-and-cow meow milk moofasa moose mutilated ren sheep skeleton small

stegosaurus stimpy supermilker surgery telebears three-eyes turkey turtle

tux udder vader vader-koala www

[root@pubserver ansible]# cowsay -f elephant are you ok?

_____________

< are you ok? >

-------------

\     /\  ___  /\

\   // \/   \/ \\

((    O O    ))

\\ /     \ //

 \/  | |  \/

  |  | |  |

  |  | |  |

  |   o   |

  | |   | |

  |m|   |m|

# 如果执行playbook时也会出现奶牛,则可以使用以下方式取消

[root@pubserver ansible]# echo 'nocows = 1' >> ansible.cfg

你可能感兴趣的:(android,服务器,运维,ansible)