• Ansible自动化运维工具之playbook剧本编写(上)


     内容预知

    目录

     内容预知

     1.playbook的相关知识

    1.1 playbook 的简介 

    1.2 playbook的 各部分组成

     2. 基础的playbook剧本编写实例

    实例1:playbook编写 apache的yum安装部署剧本

    实例2:playbook编写nginx 的yum安装并且能修改其监听端口的剧本

     3. playbook的定义、引用变量

     3.1 基础变量的定义与引用

    3.2 引用fact信息中的变量 

     4. playbook中的when条件判断和变量循环使用

     4.1 when条件判断

    4.2 变量循环

      (1)with_item 单循环输出

    (2) with_list  每组列表一起循环的输出

    (3) with_together 同一列表位置数据组合输出的循环

    (4) with_nested 列表数据循环匹配的循环(根据列表个数定义有多少层的循环)

    四种迭代循环方式的总结

     5. Templates 模块

    (1)先准备一个以 .j2 为后缀的 template 模板文件,设置引用的变量 

    (2) 修改主机清单文件,使用主机变量定义一个变量名相同,而值不同的变量

    (3) 编写 playbook 

     6. Tags

    6.1 单标签的使用

    6.2 多标签的运用

    6.3 通用标签always的运用 

     7. roles


     1.playbook的相关知识

    1.1 playbook 的简介 

    playbook是 一个不同于使用Ansible命令行执行方式的模式,其功能更强大灵活。简单来说,playbook是一个非常简单的配置管理和多主机部署系统,不同于任何已经存在的模式,可作为一个适合部署复杂应用程序的基础。Playbook可以定制配置,可以按照指定的操作步骤有序执行,支持同步和异步方式。我们完成一个任务,例如安装部署一个httpd服务,我们需要多个模块(一个模块也可以称之为task)提供功能来完成。而playbook就是组织多个task的容器,他的实质就是一个文件,有着特定的组织格式,它采用的语法格式是YAML(Yet Another Markup Language)。

    对于YAML文件格式的编写,我在之前的博客就有所介绍:

    【云原生】Docker-compose单机容器集群编排_站在这别动,我去给你买橘子的博客-CSDN博客https://blog.csdn.net/qq_62462797/article/details/128052363?spm=1001.2014.3001.5501
     

    1.2 playbook的 各部分组成

    (1)Tasks:任务,即通过 task 调用 ansible 的模板将多个操作组织在一个 playbook 中运行
    (2)Variables:变量
    (3)Templates:模板
    (4)Handlers:处理器,当changed状态条件满足时,(notify)触发执行的操作
    (5)Roles:角色

     

     2. 基础的playbook剧本编写实例

    playbook中运用的模块就是ansible中的模块,就像docker-compose一样将docker操作容器的指令归纳为一个yaml文件,开启运行yaml中的指令模块就能按照预设计的方向去完成。 

    实例1:playbook编写 apache的yum安装部署剧本

    剧本编写实现的需求对Ansible管理的所有的webservers组的成员,yum安装最新版本的apache服务软件,并进行相应环境的调整,确保webservers的apache服务能够正常运行并设置开机自启 

    1. cd /etc/ansible #在ansible的所在目录中创建该项目的目录
    2. mkdir apache
    3. vim apache.yaml
    4. ---
    5. - name: apache yum apply
    6. gather_facts: false
    7. hosts: webservers
    8. remote_user: root
    9. tasks:
    10. - name: test connection
    11. ping:
    12. - name: stop firewalld
    13. service: name=firewalld state=stopped
    14. - name: stop selinux
    15. command: '/usr/sbin/setenforce 0'
    16. ignore_errors: true
    17. - name: yum install apache service
    18. yum: name=httpd state=latest
    19. - name: start apache service
    20. service: name=httpd state=started enabled=yes

     

    运行剧本的方法:

    1. //运行playbook
    2. ansible-playbook apache.yaml
    3. //补充参数:
    4. -k(–ask-pass):用来交互输入ssh密码
    5. -K(-ask-become-pass):用来交互输入sudo密码
    6. -u:指定用户
    7. ansible-playbook apache.yaml --syntax-check #检查yaml文件的语法是否正确
    8. ansible-playbook apache.yaml --list-task #检查tasks任务
    9. ansible-playbook apache.yaml --list-hosts #检查生效的主机
    10. ansible-playbook apache.yaml --start-at-task='install httpd' #指定从某个task开始运行

     

    实例2:playbook编写nginx 的yum安装并且能修改其监听端口的剧本

     需求:通过yum安装nginx服务,并且能够控制被管理的主机的服务的开启,按照预设的配置在运行时的端口。

    在编写剧本前,需要准备相应的两个文件,一个为nginx的yum源。一个为相对应的主配置文件,在主配置文件中修改其端口,在将该配置移至被管理主机中,作为运行启动时的默认配置 

    剧本编写:

    1. mkdir /etc/ansible/nginx
    2. vim nginx.yaml
    3. ---
    4. - name: nginx script
    5. gather_facts: false
    6. hosts: webservers
    7. remote_user: root
    8. tasks:
    9. - name: test connection
    10. ping:
    11. - name: stop firewalld
    12. service: name=firewalld state=stopped enabled=no
    13. - name: stop selinux
    14. command: '/usr/sbin/setenforce 0'
    15. ignore_errors: true
    16. - name: prepare nginx repo
    17. copy: src=/etc/ansible/nginx/nginx.repo dest=/etc/yum.repos.d/nginx.repo
    18. - name: install nginx
    19. yum: name=nginx state=latest
    20. - name: change port
    21. copy: src=/opt/default.conf dest=/etc/nginx/conf.d/default.conf
    22. notify: "restart nginx"
    23. - name: start nginx
    24. service: name=nginx state=started enabled=yes
    25. handlers:
    26. - name: restart nginx
    27. service: name=nginx state=restarted

     

    运行结果:

     

     

     3. playbook的定义、引用变量

     3.1 基础变量的定义与引用

    在yaml文件中,我们可以在初始配置的模块中用var去定义变量的存在,变量的格式为key:value,以此来确定该变量在剧本中的存在 

    1. vim test1.yaml
    2. ---
    3. - name: this is a play for testing variables
    4. hosts: dbservers
    5. remote_user: root
    6. vars:
    7. filename: abc.txt
    8. tasks:
    9. - name: touch a test file
    10. file: path=/opt/{{filename}} state=touch
    11. ansible-playbook test1.yaml

    3.2 引用fact信息中的变量 

    首先我们知道  使用 ansible 组  -m setup   可以收集该组中所有的节点信息 ,

    所以setup中fact'信息,有时候会剧本编写中需要,而fact的信息也是可以通过变量的方式进行调用

     

    剧本编写:

    1. vim test2.yaml
    2. ---
    3. - name: this is a playbook for quote variate
    4. hosts: dbservers
    5. remote_user: root
    6. tasks:
    7. - name: reading setup fact variate
    8. debug: msg={{ansible_date_time.weekday}}
    9. ~

    运行的结果:
     

     4. playbook中的when条件判断和变量循环使用

     4.1 when条件判断

    1. #选用filter=ansible_default_ipv4中的address作为when条件进行测试
    2. ansible all -m setup -a 'filter=ansible_default_ipv4'

    测试剧本编写:

    1. vim test3.yaml
    2. ---
    3. - name: this is when test playbook
    4. hosts: all
    5. remote_user: root
    6. tasks:
    7. - name: test when
    8. debug: msg='判断位置'
    9. when: ansible_default_ipv4.address == "192.168.73.107"
    10. ansible-playbook test3.yaml

     

    除此之外 when条件还可以通过 !=(不等于条件来进行判断) 

    1. vim test3.yaml
    2. ---
    3. - name: this is when test playbook
    4. hosts: all
    5. remote_user: root
    6. tasks:
    7. - name: test when
    8. debug: msg='判断位置'
    9. when: ansible_default_ipv4.address != "192.168.73.107"
    10. ansible-playbook test3.yaml

    4.2 变量循环

      (1)with_item 单循环输出

    1. vim test4.yaml
    2. ---
    3. - name: item test
    4. hosts: dbservers
    5. remote_user: root
    6. gather_facts: no
    7. tasks:
    8. - debug:
    9. msg: "{{item}}"
    10. with_items: [a, b, c, d]
    11. ansible-playbook test4.yaml

     

     当列表为两个时。with_item的输出方式:

    1. vim test4.yaml
    2. ---
    3. - name: item test
    4. hosts: dbservers
    5. remote_user: root
    6. gather_facts: no
    7. tasks:
    8. - debug:
    9. msg: "{{item}}"
    10. with_items:
    11. - [a, b, c, d]
    12. - [1 ,2, 3, 4]
    13. ansible-playbook test4.yaml

    (2) with_list  每组列表一起循环的输出

    1. ---
    2. - name: item test
    3. hosts: dbservers
    4. remote_user: root
    5. gather_facts: no
    6. tasks:
    7. - debug:
    8. msg: "{{item}}"
    9. with_list:
    10. - [a, b, c, d]
    11. - [1 ,2, 3, 4]
    12. ~
    13. ~

    (3) with_together 同一列表位置数据组合输出的循环

    1. ---
    2. - name: item test
    3. hosts: dbservers
    4. remote_user: root
    5. gather_facts: no
    6. tasks:
    7. - debug:
    8. msg: "{{item}}"
    9. with_together:
    10. - [a, b, c, d]
    11. - [1 ,2, 3, 4]
    12. ~

    1. ---
    2. - name: item test
    3. hosts: dbservers
    4. remote_user: root
    5. gather_facts: no
    6. tasks:
    7. - debug:
    8. msg: "{{item}}"
    9. with_together:
    10. - [a, b, c, d]
    11. - [1 ,2, 3, 4]
    12. - [A, B, C]

     

    (4) with_nested 列表数据循环匹配的循环(根据列表个数定义有多少层的循环)

    1. ---
    2. - name: item test
    3. hosts: dbservers
    4. remote_user: root
    5. gather_facts: no
    6. tasks:
    7. - debug:
    8. msg: "{{item}}"
    9. with_nested:
    10. - [a, b, c, d]
    11. - [1 ,2, 3, 4]
    12. ~

     

    四种迭代循环方式的总结

     whith_items:  {{item}}会把所有的列表展开进行遍历输出,with_flattened也可以替代with_items

     with_list:    {{item}}会把每个列表当作一个整体输出。如果每个列表中只有一个值,则效果与with items一致。loop也可以替代ith

     with_together: {{item}}引用时会把每个列表相同位置的值对齐合并后输出

    with nested:{ {item}}引用时会把每个列表的值两两组合循环输出

     5. Templates 模块

    Jinja是基于Python的模板引擎。Template类是Jinja的一个重要组件,可以看作是一个编译过的模板文件,用来产生目标文本,传递Python的变量给模板去替换模板中的标记。 

     本次我们以改变apche的配置文件为例,来展现Templates模块的运用

    (1)先准备一个以 .j2 为后缀的 template 模板文件,设置引用的变量 

    1. #如果没有相关的httpd的配置文件,可以先yum按住一个httpd的服务,取其主配置文件
    2. cp /etc/httpd/conf/httpd.conf /opt/httpd.conf.j2
    3. vim /opt/httpd.conf.j2
    4. Listen {{http_port}} #42行,修改
    5. ServerName {{server_name}} #95行,修改
    6. DocumentRoot "{{root_dir}}" #119行,修改

    (2) 修改主机清单文件,使用主机变量定义一个变量名相同,而值不同的变量

    1. vim /etc/ansible/hosts
    2. [webservers]
    3. 192.168.73.106 http_port=192.168.73.106:80 server_name=www.test1.com:80 root_dir=/etc/httpd/htdocs
    4. [dbservers]
    5. 192.168.73.107 http_port=192.168.73.107:80 server_name=www.test2.com:80 root_dir=/etc/httpd/htdocs

     

    此外如果没有做DNS解析域名,还需要对主机名进行映射 :

    1. vim /etc/hosts
    2. 192.168.73.106 www.test1.com
    3. 192.168.73.107 www.test2.com

    (3) 编写 playbook 

    1. mkdir /etc/ansible/templates
    2. vim apache.yaml
    3. ---
    4. - hosts: all
    5. remote_user: root
    6. vars:
    7. - package: httpd
    8. - service: httpd
    9. tasks:
    10. - name: install httpd package
    11. yum: name={{package}} state=latest
    12. - name: install configure file
    13. template: src=/opt/httpd.conf.j2 dest=/etc/httpd/conf/httpd.conf
    14. notify:
    15. - restart httpd
    16. - name: create root dir
    17. file: path=/etc/httpd/htdocs state=directory
    18. - name: start httpd server
    19. service: name={{service}} enabled=true state=started
    20. handlers:
    21. - name: restart httpd
    22. service: name={{service}} state=restarted
    23. ansiable-playbook apache.yaml

     

     

     

     6. Tags

     可以在一个playbook中为某个或某些任务定义“标签”,在执行此playbook时通过ansible-playbook命令使用--tags选项能实现仅运行指定的tasks。
    playbook还提供了一个特殊的tags为always。作用就是当使用always作为tags的task时,无论执行哪一个tags时,定义有always的tags都会执行。

    6.1 单标签的使用

    1. vim test1.yaml
    2. ---
    3. - name: this is a play for testing variables
    4. hosts: dbservers
    5. remote_user: root
    6. vars:
    7. filename: abc.txt
    8. tasks:
    9. - name: position 1
    10. debug:
    11. msg: 'ls /opt'
    12. tags:
    13. - only
    14. - name: position 2
    15. debug:
    16. msg: 'ls /mnt'
    17. ansible-playbook test1.yaml --tags="only"

    6.2 多标签的运用

    1. ---
    2. - name: this is a play for testing variables
    3. hosts: dbservers
    4. remote_user: root
    5. vars:
    6. filename: abc.txt
    7. tasks:
    8. - name: position 1
    9. debug:
    10. msg: '测试标签1'
    11. tags:
    12. - one
    13. - name: position 2
    14. debug:
    15. msg: '测试标签2'
    16. tags:
    17. - two
    18. - name: position 3
    19. debug:
    20. msg: '测试标签3'
    21. tags:
    22. - one

    执行结果:

     

    6.3 通用标签always的运用 

    1. ---
    2. - name: this is a play for testing variables
    3. hosts: dbservers
    4. remote_user: root
    5. vars:
    6. filename: abc.txt
    7. tasks:
    8. - name: position 1
    9. debug:
    10. msg: '测试标签1'
    11. tags:
    12. - one
    13. - name: position 2
    14. debug:
    15. msg: '测试通用标签always'
    16. tags:
    17. - always
    18. - name: position 3
    19. debug:
    20. msg: '测试标签3'
    21. tags:
    22. - one

     

    执行结果:

     

     7. roles

    • Roles又称为角色,playbook被称为剧本。Roles角色是自1.2版本之后引入的新特性,用于层次性、结构化的组织剧本
       
    • roles能够根据层次型结构自动装载变量文件、任务集、以及触发的动作等,要使用roles只需要在剧本中使用include命令引入即可
    • 简单的来说,roles就是分别将变量、文件、任务、模板以及处理器放置于不同的单独的目录,并且可以便捷的通过include引入
    • 角色一般用于基于主机构建的服务的场景中,但是也可以是用于构建守护进程等场景中,主要是使用在代码复用度较高的场景下

    ●files
    用来存放由 copy 模块或 script 模块调用的文件。

    ●templates
    用来存放 jinjia2 模板,template 模块会自动在此目录中寻找 jinjia2 模板文件。

    ●tasks
    此目录应当包含一个 main.yml 文件,用于定义此角色的任务列表,此文件可以使用 include 包含其它的位于此目录的 task 文件。

    ●handlers
    此目录应当包含一个 main.yml 文件,用于定义此角色中触发条件时执行的动作。

    ●vars
    此目录应当包含一个 main.yml 文件,用于定义此角色用到的变量。

    ●defaults
    此目录应当包含一个 main.yml 文件,用于为当前角色设定默认变量。

    ●meta
    此目录应当包含一个 main.yml 文件,用于定义此角色的特殊设定及其依赖关系。

     

    集中式lamp的简单role编写过程:

    1. //在一个 playbook 中使用 roles 的步骤:
    2. 1)创建以 roles 命名的目录
    3. mkdir /etc/ansible/roles/ -p #yum装完默认就有
    4. 2)创建全局变量目录(可选)
    5. mkdir /etc/ansible/group_vars/ -p
    6. touch /etc/ansible/group_vars/all #文件名自己定义,引用的时候注意
    7. 3)在 roles 目录中分别创建以各角色名称命名的目录,如 httpd、mysql
    8. mkdir /etc/ansible/roles/httpd
    9. mkdir /etc/ansible/roles/mysql
    10. 4)在每个角色命名的目录中分别创建files、handlers、tasks、templates、meta、defaults和vars目录,用不到的目录可以创建为空目录,也可以不创建
    11. mkdir /etc/ansible/roles/httpd/{files,templates,tasks,handlers,vars,defaults,meta}
    12. mkdir /etc/ansible/roles/mysql/{files,templates,tasks,handlers,vars,defaults,meta}
    13. 5)在每个角色的 handlers、tasks、meta、defaults、vars 目录下创建 main.yml 文件,千万不能自定义文件名
    14. touch /etc/ansible/roles/httpd/{defaults,vars,tasks,meta,handlers}/main.yml
    15. touch /etc/ansible/roles/mysql/{defaults,vars,tasks,meta,handlers}/main.yml
    16. 6)修改 site.yml 文件,针对不同主机去调用不同的角色
    17. vim /etc/ansible/site.yml
    18. ---
    19. - hosts: webservers
    20. remote_user: root
    21. roles:
    22. - httpd
    23. - hosts: dbservers
    24. remote_user: root
    25. roles:
    26. - mysql
    27. 7)运行 ansible-playbook
    28. cd /etc/ansible
    29. ansible-playbook site.yml
    30. 示例:
    31. mkdir /etc/ansible/roles/httpd/{files,templates,tasks,handlers,vars,defaults,meta} -p
    32. mkdir /etc/ansible/roles/mysql/{files,templates,tasks,handlers,vars,defaults,meta} -p
    33. mkdir /etc/ansible/roles/php/{files,templates,tasks,handlers,vars,defaults,meta} -p
    34. touch /etc/ansible/roles/httpd/{defaults,vars,tasks,meta,handlers}/main.yml
    35. touch /etc/ansible/roles/mysql/{defaults,vars,tasks,meta,handlers}/main.yml
    36. touch /etc/ansible/roles/php/{defaults,vars,tasks,meta,handlers}/main.yml
    37. ------编写httpd模块------
    38. 写一个简单的tasks/main.yml
    39. vim /etc/ansible/roles/httpd/tasks/main.yml
    40. - name: install apache
    41. yum: name={{pkg}} state=latest
    42. - name: start apache
    43. service: enabled=true name={{svc}} state=started
    44. //定义变量:可以定义在全局变量中,也可以定义在roles角色变量中,一般定义在角色变量中
    45. vim /etc/ansible/roles/httpd/vars/main.yml
    46. pkg: httpd
    47. svc: httpd
    48. -------编写mysql模块-------
    49. vim /etc/ansible/roles/mysql/tasks/main.yml
    50. - name: install mysql
    51. yum: name={{pkg}} state=latest
    52. - name: start mysql
    53. service: enabled=true name={{svc}} state=started
    54. vim /etc/ansible/roles/mysql/vars/main.yml
    55. pkg:
    56. - mariadb
    57. - mariadb-server
    58. svc: mariadb
    59. -------编写php模块-----
    60. vim /etc/ansible/roles/php/tasks/main.yml
    61. - name: install php
    62. yum: name={{pkg}} state=latest
    63. - name: start php-fpm
    64. service: enabled=true name={{svc}} state=started
    65. vim /etc/ansible/roles/php/vars/main.yml
    66. pkg:
    67. - php
    68. - php-fpm
    69. svc: php-fpm
    70. -----编写roles示例-----
    71. vim /etc/ansible/site.yml
    72. ---
    73. - hosts: webservers
    74. remote_user: root
    75. roles:
    76. - httpd
    77. - mysql
    78. - php
    79. cd /etc/ansible
    80. ansible-playbook site.yml

  • 相关阅读:
    C++ 多态详解【用法+原理】
    【C语言】简单实现扫雷游戏
    MySQL-字符串按照数值排序
    国家网信办发布第十三批境内区块链信息服务备案编号
    汽车产业互联网如何跟上“智能”和“市场”节奏
    Vue 简单的语法
    高性能图表组件LightningChart .NET v11.0发布——增强DPI感知能力
    无人零售与传统便利店的竞争优势
    buildadmin+tp8表格操作(6.1)表格行中添加详情链接并弹出对话框
    【剑指 Offer】矩阵中的路径
  • 原文地址:https://blog.csdn.net/qq_62462797/article/details/128160646