243张图片为你解析Linux轻量级自动运维化工具Ansible

Ansible 是什么 ?

 

v2-dbf3cd07640fc1d2822fb29ba93781a1_hd.jpg

 

ansible架构图

 

v2-53195b59ff845292af5da2fc6d248e43_hd.jpg

 

ansible特性

模块化:调用特定的模块,完成特定的任务;

基于Python语言研发,由Paramiko, PyYAML和Jinja2三个核心库实现;

部署简单:agentless;

支持自定义模块,使用任意编程语言;

强大的playbook机制;

幂等性;

安装及程序环境:

程序:

ansible

ansible-playbook

ansible-doc

配置文件:

/etc/ansible/ansible.cfg

主机清单:

/etc/ansible/hosts

插件目录:

/usr/share/ansible_plugins/

安装ansible

 

v2-ad7105f785670b3959933ab1791e99bb_hd.jpg

 

安装依赖包

 

v2-effeccdce5d2f1b63b899a76c34e59b3_hd.jpg

 

ansible命令的使用:

Usage: ansible <host-pattern> [options]

常用选项:

-m MOD_NAME

-a MOD_ARGS

配置Host Inventory:

/etc/ansible/hosts

[group_id]

HOST_PATTERN1

HOST_PATTERN2

示例:

首先对此文件进行备份操作,以防后面需要用到默认配置文件

 

v2-52904fa61f3b18697c2c4cf971ac60f7_hd.jpg

 

进入到/etc/ansible/hosts文件,此处绿色光标以下的内容是没有用的,都是示例,可以删除掉,然后添加我们下面实验操作用到的主机。

 

v2-0382a4e2a687450a67cfc9ce92cd0563_hd.jpg

 

添加一组websrvs服务器,以用于下面的测试

 

v2-83d5a28e303a9eea14d9f92366729706_hd.jpg

 

测试主机连通性

这里报错是因为实验用的主机交换其他两台主机的公钥/私钥的原因导致的

 

v2-845bdc3d3600109bd9e7f88b61f25d4d_hd.jpg

 

实验SSH免密码登陆设置

生成私钥和公钥 ssh-keygen -t rsa -P ''

 

v2-5c0a0e41431157a40b6da1d2a1930e61_hd.jpg

 

复制公钥文件问authorized_keys

 

v2-469239dfeae278fb9cb715630981800f_hd.jpg

 

把公钥传送到其他主机

 

v2-fe05d076c4cf504bbd95654eb2f0e475_hd.jpg

 

在68的主机上面可以看见公钥已经传送过来了,并且确认文件的权限是否正确

 

v2-ad495226607d2fe17aa39a280b68eac6_hd.jpg

 

重复以上操作把公钥发送给69的主机

 

v2-2d392930b7f724831f7bbda9f101c2fc_hd.png

 

然后重新执行ansible的ping模块命令查看该两台主机的连通性

可以发现此时已经成功,那么下面就开始介绍ansilbe的其他模块

 

v2-7a035d28c51aaa99a08f1fe94ad34506_hd.jpg

 

最后记得利用ansible同步一下所有主机的时间,以免某主机的时间有错误,后面看日志起来会造成混乱

 

v2-7db8a0838c3ae1a18fb94e368a6e86b8_hd.jpg

 

ansible模块:

获取模块列表:ansible-doc -l

获取指定模块的使用帮助:ansible-doc -s MOD_NAME

常用模块:

ping模块:探测目标主机是否存活;

 

v2-06e9afa82fe9b4a12d34659b5ba68dae_hd.jpg

 

示例:测试所有的主机的连通性

 

v2-9d5d92197695380a9ece8d752405bcb8_hd.jpg

 

command模块:在远程主机执行命令;

 

v2-9daa9a58e54beb813ae19dc7e15fb2a4_hd.jpg

 

示例1:让所有主机同步时间

此处没有给出指定的-m command命令,是因为ansible的模块默认就是command

 

v2-9daa9a58e54beb813ae19dc7e15fb2a4_hd.jpg

 

示例2:让每一台主机都执行uname -r命令

 

v2-75b8dd17817ce5766f5a00674890c9b9_hd.jpg

示例3:在主机上面都创建一个用户

 

v2-b6e840c100602ce50c9cff0fda68da25_hd.jpg

 

查看两台主机是否已经创建该用户

 

v2-b47c52f8dbf09dbd5b65d5b1dfa47254_hd.jpg

 

 

v2-15aa24b8e49d4a374fc0d0aebc6b25e8_hd.jpg

 

 

v2-caff64e27fb57892469de9f2c5ea9d0e_hd.jpg

查看用户信息:

 

v2-aaf3d5439341097e3c8ee811245bad68_hd.jpg

 

帮这两个用户改密码,此处需要注意的是,虽然用下面的命令看似执行成功,但是当我们验证的时候,就会发现密码错误了,这是因为ansible的command模块并不支持管道等输出,所以下面介绍另外一个ansible的模块shell

 

v2-6a590fd53ee91db42612d99516bcba9b_hd.jpg

 

 

v2-8f851b670d41ecb440ec80a5d2369862_hd.jpg

 

shell模块 :在远程主机上调用shell解释器运行命令,支持shell的各种功能,例如管道等

注意:command和shell模块的核心参数直接为命令本身;而其它模块的参数通常为“key=value”格式;

 

v2-51870fbe8a2dc62cf578cd2faeb071ab_hd.jpg

 

示例:批量修改其他主机的特定用户的密码

 

v2-dcb5543d462b4424659517ad68cbd9a2_hd.jpg

 

此时可以发现已经可以登陆成功

 

v2-3dfc3df493790ab7aaff5a28ab1edd05_hd.jpg

 

copy模块:复制文件到远程主机

用法:

(1) 复制文件

-a "src='#'" "

(2) 给定内容生成文件

-a "content= dest= "

其它参数:mode, owner, group, ...

 

v2-316d7ea6185ec4f47f2944d576098623_hd.jpg

 

示例:复制文件到其他主机

此处创建一个测试文件

 

v2-fcecc387613dc6466d3450e33f840d05_hd.jpg

 

复制文件到其他主机

下面红色的报错信息是,如果要传送文件,该主机的指定目录需要存在,如果不存在,就是提示错误

 

v2-38587fb4199cb513c9108d5e5d5652bc_hd.jpg

 

创建对应的目录

 

v2-9f12ae127ee6ab64d27bf655d4407806_hd.jpg

 

重新传送文件,已经没有错误提示,但是此处也可以看见,如果文件已经存在,则原文件会被覆盖掉,并且此处也没有任何提示覆盖文件的信息,所以操作的时候就需要注意了,以免覆盖掉重要的文件

 

v2-af8a68d81b4acb40d7618a7689232ddb_hd.jpg

 

验证文件

 

v2-3cc5b1ec5808014f621804d0e891d8fe_hd.jpg

 

file模块:设置文件的属性

用法:

(1) 创建目录:

-a "path= state=directory"

(2) 创建链接文件:

-a "path= src='#'" /p>

(3) 删除文件:

-a "path= state=absent“

 

v2-b202b600512011b4c0b7415f46ebdc40_hd.jpg

 

示例:修改文件的权限和属主

 

v2-dec0b2d0182cff806173f77bb0561923_hd.jpg

 

验证文件

 

v2-8dfeec521789a9d996315212df675fc2_hd.jpg

 

示例:创建文件的软连接

 

v2-9c980af26460858c7dae369ccb18f57b_hd.jpg

 

验证文件

 

v2-5f6fac5da760322c5c1af47af29f54de_hd.jpg

 

设置文件的状态为absent(即删除文件)

 

v2-a7dd88b8d82a90da6526bd42078b6900_hd.jpg

 

验证

 

v2-8d48a91ff287235c36cdf777c199d64a_hd.jpg

 

fetch模块:从远程主机拿文件

 

v2-fe9eb7c26618dded032b95e5e85e9451_hd.jpg

 

示例:从10.1.156.69主机拿一个文件

 

v2-851835900ef13b6dfb444175f113e318_hd.jpg

 

当抓去一堆文件的时候,也会创建对应的ip地址的目录,以区分文件

 

v2-a38a38b6ba9b14dd90a5ba923d59aacc_hd.jpg

 

cron模块 :管理计划任务条目

用法:

-a ""

minute=

hour=

day=

month=

weekday=

job=

name=

user=

state={present|absent}

 

v2-39d3dbb1b3feb1b445a30576c6df165b_hd.jpg

 

示例:创建一个同步时间的计划任务,每5分钟同步一下服务器的时间

 

v2-9d8ff378b60e73a47d6b38a85f5baa00_hd.jpg

 

验证任务

 

v2-d764d60250e64806afbc4a359219ed2e_hd.jpg

 

示例:删除计划任务

 

v2-792ab77141020e575c627aac9dfaf3ae_hd.jpg

 

验证

 

v2-6906ac71fd62421c239d360c663a3c4b_hd.jpg

 

hostname模块:管理主机名

用法:

name=

 

v2-6efdbd0c7c103b5eb905aa8db48008fd_hd.jpg

 

示例:修改主机名

 

v2-3d7412c5a2ae52edd42e87d87e537014_hd.jpg

 

 

v2-7892fc61fc0fd29827470261633223d7_hd.jpg

 

yum模块:使用yum命令完成程序包管理

用法:

-a ""

(1) name= state={present|latest}

(2) name= state=absent

 

v2-b880e5a4241ca382cf087f00d47d0361_hd.jpg

 

示例:安装指定包

此实验,首先,确定主机的yum源是可用的,否则实验会失败

 

v2-a36d7d88daf145897f22459daf0326c8_hd.jpg

 

安装samba包

 

v2-23e7de0926951cbad2e975e733b8d73e_hd.jpg

 

验证

 

v2-f2c1d1a97ab5f8f52f7e46f723c9ae8f_hd.jpg

 

删除samba安装包

 

v2-2ff439253e6e8c8ee0a09d30d48f36bb_hd.jpg

 

已经没有安装的字眼了

service模块:服务管理

用法:

-a ""

name=

state=

started

stopped

restarted

enabled=

runlevel=

 

v2-055ae4b999abbf7e5fe5988d550f0ee1_hd.jpg

 

示例:开启主机的httpd服务

首先我们确认httpd服务是关闭的

 

v2-826537c9a882dfe8fa2481aa066aa95d_hd.jpg

 

开启httpd服务,并且设置为开机启动

 

v2-ee40eee87424cbc1b2568f428f245d58_hd.jpg

 

验证,80端口已经开启

 

v2-2c4a5c769e427001f6a152b32f1143d0_hd.jpg

 

group模块:增加或删除组

用法:

-a ""

name=

state=

system=

gid=

 

v2-bf8fec24590a5f8d6ee8211bb76cc7ae_hd.jpg

 

示例:添加一个组

 

v2-d280cc8c3282e74557309c03a4d2c71e_hd.jpg

 

验证

 

v2-e9a7f8ce1622c19e3a7b47724acc9c88_hd.jpg

 

删除组

 

v2-ac2d2a2ff407efb4fd9b536c0bef7fc5_hd.jpg

 

 

v2-97fcc12e7ca55637a649421ff275e546_hd.jpg

 

验证

 

v2-69a675c7c6d85c2d3b13ac60e32db4fe_hd.jpg

 

user模块:用户管理

使用格式:

name= : 创建的用户名

state= : present新增,absent删除

force= : 删除用户的时候删除家目录

system= : 创建系统用户

uid= : 指定UID

shell= : 指定shell

home= : 指定用户家目录

 

v2-2d13f1f70b6955760b1880f03746ccd1_hd.jpg

 

示例:增加一个系统用户

 

v2-93011de816e43169a02225672f0a87da_hd.jpg

 

验证

 

v2-1fd1cff974a3c3fbe2a62b6de8d07ad0_hd.jpg

 

删除用户

 

v2-b8e7e9b9ea7cb2445526bd03196be729_hd.jpg

 

setup模块:收集主机里面的各种信息

 

v2-41272c09b46384e352777429fe379f61_hd.jpg

 

示例:收集所有主机的信息

 

v2-cd6058e50d2eaac07ce99f5b0a929e34_hd.jpg

 

YAML  一种数据序列化工具的语言格式

YAML is a data serialization format designed for human readability and interaction with scripting languages.

 

v2-f1811b291f0c97437a719bde089dfe96_hd.jpg

 

数据结构:

key:value

- item1

- item2

- item3

例如{name:jerry, age:21}

PlayBook

核心元素:

Tasks:任务,由模块定义的操作的列表;

Variables:变量

Templates:模板,即使用了模板语法的文本文件;

Handlers:由特定条件触发的Tasks;

Roles:角色;

playbook的基础组件:

Hosts:运行指定任务的目标主机;

remote_user:在远程主机以哪个用户身份执行;

sudo_user:非管理员需要拥有sudo权限;

tasks:任务列表

模块,模块参数:

格式:

(1) action: module arguments

(2) module: arguments

运行playbook,使用ansible-playbook命令

(1) 检测语法

ansible-playbook --syntax-check /path/to/playbook.yaml

(2) 测试运行

ansible-playbook -C /path/to/playbook.yaml

--list-hosts

-list-tasks

--list-tags

(3) 运行

ansible-playbook /path/to/playbook.yaml

-t TAGS, --tags=TAGS

--skip-tags=SKIP_TAGS

--start-at-task=START_AT

示例1:定义一个playbook任务来新增用户和组

定义一个yaml的模板

 

v2-a19ad7a0412267b2c9b2c1208296de3c_hd.jpg

 

 

v2-ace52e3c6ca96776294f7a3b5377eaa0_hd.jpg

 

查查语法有没有错误,没有提示即表示语法应该没有问题。

 

v2-6635e17a5fb56dd2ec53c36a855bd0f8_hd.jpg

 

测试运行看看,-C表示仅测试跑一边,但是不会实际操作

 

v2-d89afd2592cba130c6c5ed74a9141a6b_hd.jpg

 

也可以单独测试某些特定的选项

查看仅影响的主机

 

v2-989ebc443bc8be62e2d6345b1be1a3a7_hd.jpg

 

查看运行哪些任务

 

v2-6396ad079770e0b35c35f9bd7a9e9f2a_hd.jpg

 

查看哪个任务打标了,这里并没有任何任务打标记,后面再演示

 

v2-adb9fce0695942ddf6af1def5562f47e_hd.jpg

 

以上没有错误,开始正式运行该任务

 

v2-5b32ff82704c29c161ed1a4d93b71d57_hd.jpg

 

验证

 

v2-5505b5c25f2a269ef7a9810b5c3ff1ee_hd.jpg

 

示例2:定义一个playbook任务来修改文件端口

 

v2-5a7e6edfd06884fd14698ee92eaf8928_hd.jpg

 

此步骤里面有安装httpd的安装包,其实此处有点多余,因为测试的两台主机均已经安装该服务,此处添加上去是为了演示效果,因为当生产环境中,假如存在一台服务器没有该安装包,那么次处就能帮我们安装上去,不然的话,漏了这一步,到后面查原因也挺麻烦的

 

v2-cca04b1700e4ceae0223aa4468d397a0_hd.jpg

 

检查语法问题

 

v2-ae6238331d17adde7ce89cadeeca665d_hd.jpg

 

先从一台主机上面把httpd.conf文件拷问来编辑

 

v2-84d83ca446f1bef7a075415a858e663e_hd.jpg

 

修改httpd.conf文件

比如修改端口为8080,其他都为默认配置

 

v2-3c979301ff379d22506b2ccd23f96b70_hd.jpg

 

首先备份好各自主机里面的配置文件,以防后面出错

 

v2-ae4a3478a93630dac3f7d4c613993ed8_hd.jpg

 

检查备份是否成功

 

v2-85797cea02e5d8f316843b9c92b092ba_hd.jpg

 

测试运行web.yml,看看有没有问题,没有问题的话就正常运行

 

v2-04e344cdf999d48fa44ae6ce1cb34782_hd.jpg

执行改文件

 

v2-932ee1154bc6a8005a10f6d5a45c93f7_hd.jpg

 

验证服务器端口打开没有,可以看见8080端口已经打开,实验成功。

 

v2-5d0859b5ed5da2e1ec8ce36ddca9a0c3_hd.jpg

 

Handlers的使用:由特定条件触发的Tasks;

格式:

tasks:

- name: TASK_NAME

module: arguments

notify: HANDLER_NAME

handlers:

- name: HANDLER_NAME

module: arguments

示例:参照上面的例子继续修改apache的端口

修改端口号为8090

 

v2-afec620dbbb6e785aa69da3c15d88396_hd.jpg

 

修改原来的web.yml脚本实现操作

 

v2-09f0f6aee86c3dc007f7a52d8df2e4d7_hd.jpg

 

检测语法

 

v2-e6d1da70c438da489324da73e9fcf6b2_hd.jpg

 

测试运行,可以看出,当复制文件过去的时候,会触发到restart httpd service的handlers任务,所以任务就重启了,而不是启动

 

v2-a7947f2ba1d44593d1a28ac2414d25d1_hd.jpg

 

正式运行

 

v2-b8e940d38ffb4c26f8054de66556e0ea_hd.jpg

 

验证结果,8090端口已经打开,实验成功

 

v2-ffccc978fe99069d116df70a4433ed5a_hd.jpg

 

tags:给指定的任务定义一个调用标识;

使用格式:

- name: NAME

module: arguments

tags: TAG_ID

示例:执行特定的tags

修改文件的端口为8088

 

v2-1a1bff9887e3521d53718c12b99f5914_hd.jpg

 

在此前的配置文件上面插入一个标签instconf

 

v2-5295768d9b398a4c74f24fc41ab76847_hd.jpg

 

检查语法

 

v2-651fc7ebcff37cae7efe2ef0d676c10d_hd.jpg

 

此处可以查看到该yml脚本有一个标签,影响着websrvs组

 

v2-a963ff86fd9e9c4a2a11db1d37d4e02d_hd.jpg

 

测试运行

 

v2-a2f70fa6a3192bcc52ced23e0da6cb1a_hd.jpg

 

正式运行一下,指定以instconf的标签运行,所以此处不会显示器其他多余的信息,包括安装httpd包和启动httpd服务

 

v2-d26ffeb22bf43de82203f99208c7a122_hd.jpg

 

验证该结果

 

v2-42cff6df279616ceb0915c9fc56b1a03_hd.jpg

 

此处也可以对同一个文件标记多个标签同时执行

 

v2-73c19758d19dbf3d2a805edb2e9ea207_hd.jpg

 

测试运行,因为此处已经安装了httpd包和文件已经复制过去,所以都是绿色,此处就演示到这里,其他步骤可以参考上面的操作

 

v2-bea4e6d41b4b370de2f7156195474b51_hd.jpg

 

Variables:变量

类型:

内建:

(1) facts

自定义:

(1) 命令行传递;

-e VAR=VALUE

(2) 在hosts Inventory中为每个主机定义专用变量值;

(a) 向不同的主机传递不同的变量 ;

IP/HOSTNAME variable_name=value

(b) 向组内的所有主机传递相同的变量 ;

[groupname:vars]

variable_name=value

(3) 在playbook中定义

vars:

- var_name: value

- var_name: value

(4) Inventory还可以使用参数:

用于定义ansible远程连接目标主机时使用的属性,而非传递给playbook的变量;

ansible_ssh_host

ansible_ssh_port

ansible_ssh_user

ansible_ssh_pass

ansible_sudo_pass

...

(5) 在角色调用时传递

roles:

- { role: ROLE_NAME, var: value, ...}

变量调用:

{{ var_name }}

示例1:利用命令行传递变量来安装不同的包

 

v2-7cfecbea214cd02d9079f2a48474df84_hd.jpg

 

此处{{ pkgname }}表示为一个变量

 

v2-c4b143924a9fe5daef65006acde79cfb_hd.jpg

 

检查一下语法,居然报错了,什么情况?仔细看了即便发现是漏了空格

 

v2-c6647ffc059023d0a173de8a6d9dfaef_hd.jpg

 

加上空格

 

v2-a8d4afc199168715832c189f2d4a8975_hd.jpg

 

再次检查,还是还是有报错的情况,各位不要慌,因为这只是因为还没有给变量赋值才会报的错,所以此处报错是很正常

 

v2-62390f79480b7eb35c6866d1f3e1abbf_hd.jpg

 

给变量赋值再跑一遍,此时就不会报错

 

v2-7c7b88d87519247538fa35f7f4e4897b_hd.jpg

 

修改一下变量,发现也是正常的,此处68因为已经安装过vsftpd所以就不会执行,所以并不会changed

 

v2-7e2b786f6e044118c248f36b13eac7f1_hd.jpg

 

示例2:在playbook中定义变量

 

v2-057a02b639db6c780e69a748bf68c532_hd.jpg

 

测试,也没有问题的

 

v2-70720d19ff6b1d62fc85a53e7f16396f_hd.jpg

 

思考?假如同时利用-e的参数传递一个变量的参数的话会怎么样?

测试结果如下,是-e传递的变量参数的优先级更高,这样的话能避免传递参数的时候,因为文本里面定义的优先级更高而出错?

 

v2-960b4b7c91e04ba1c4cbaa9086685e02_hd.jpg

 

示例3:在hosts Inventory中为每个主机定义专用变量值

 

v2-05f99a2060bc1495a9ca7a880afd60b8_hd.jpg

 

 

v2-d0aa229eaecd0dd2609220042c4333e8_hd.jpg

 

删除掉文档里面原有的变量

 

v2-37c749e2d2e02a1051647e8af63614b6_hd.jpg

 

测试,没有问题

 

v2-a30d518a0d040f762d25916424428492_hd.jpg

 

示例4:在hosts Inventory中为每个主机定义专用变量值的第二种方法

 

v2-05f99a2060bc1495a9ca7a880afd60b8_hd.jpg

 

 

v2-dd84555193bff4392b9588b8faca4e30_hd.jpg

 

测试,也是可以的

 

v2-3be0038c153f17acdc6e27ff3a2dfc8d_hd.jpg

 

Templates :模板,文本文件,内部嵌套有模板语言脚本(使用Jinja2模板语言编写)

 

v2-e3e7e36342e61f04cca92b595424fa82_hd.jpg

 

Jinja2 is a template engine written in pure Python. It provides a Django inspired non-XML syntax but supports inline expressions and an optional sandboxed environment.

 

v2-eae5fd2738074bb5c0a3bf8e39764ac4_hd.jpg

 

语法:

字面量:

字符串:使用单引号或双引号;

数字:整数、浮点数;

列表:[item1, item2, ...]

元组:(item1, item2, ...)

字典:{key1:value1, key2:value2, ...}

布尔型:true/false

算术运算:

+, -, *, /, //, %, **

比较操作:

==, !=, >, <, >=, <=

逻辑运算:and, or, not

执行模板文件中的脚本,并生成结果数据流,需要使用template模块;

template:

-a " "

src=

dest=

mode=

onwer=

group=

注意:此模板不能在命令行使用,而只能用于playbook;

示例:利用templates模板来设置nginx的定义cpu的数量

首先利用ansible命令获取当前系统系统的cpu数量

 

v2-dff36d1f6f26e3274b5bd7ef061362a3_hd.jpg

 

首先备份一下默认的文件

 

v2-adbd341962df784deeea40eb52ebaaaf_hd.jpg

 

首先在下面的主机传送一个配置文件过来

 

v2-93bd25e9608c48cbcba9300e316c0a0e_hd.jpg

 

编辑该文件,修改此处为上面利用ansible的setup模块获取的名称

重命名该文件为Jinja2格式后缀的文件

 

v2-9d00cef2433ffe093b7d8763b339425d_hd.jpg

 

新建一个playbook文件,为了演示,建立一个ngxsrvs组,虽然看上去都一样。。。

 

v2-ae9dd337098d3b4003ba4fa01720cbdd_hd.jpg

 

建立playbook文档

 

v2-a4c81acfb9b755d08212c5a13c05f564_hd.jpg

 

 

v2-fcf3df5c923f60a13f15975984886a16_hd.jpg

 

此处为了演示效果,此处把原来的nginx包卸载掉

 

v2-a60fce2284de063a4a8d90309297c342_hd.jpg

 

确认安装包卸载掉,并且服务没在线

 

v2-7f551037fc9fced3817bcefe86ca8781_hd.jpg

 

检查playbook的文件有没语法错误

 

v2-766867d66ed4694beb263d7470b802a4_hd.jpg

 

测试运行,此处报错是因为找不到nginx的服务,所以应该是没有问题的

 

v2-2bd54f8df2695d758335166e5ec4cd24_hd.jpg

 

正式运行,没有问题

 

v2-4cb89fcefd895b1987f8a320710eab97_hd.jpg

 

查看一下端口是否已经打开

 

v2-1416f3eb335151874ea70afbd69c3b2a_hd.jpg

 

重点检查一下cpu的变量是否有改变,这里可以看到,跟我们ansible_processor_vcpus的值是一样,这样符合我们预期,此处就展示完毕

 

v2-6c5be88f94ed6018ab1c1f2af785128a_hd.jpg

 

 

v2-4d21fea504321426052a3348d971be06_hd.jpg

 

 

条件测试 :when语句:在tasks中使用,Jinja2的语法格式;

示例:利用Ansible条件测试在CentOS_6和CentOS_7的启动服务

这边首先增加一台ip为10.1.156.70的CentOS7的主机

 

v2-8e759aed7a0d985dd78f4f2784e3b1c6_hd.jpg

 

然后我们利用setup模块的命令

 

v2-cec2c584f70799a6a106189626928d77_hd.jpg

 

在7的上面可以找到该行

 

v2-fe56f9cd98260a6e433b5fd120950158_hd.jpg

 

在6的上面可以找到该行

 

v2-0de60de078e308e0a362447f3e4b1895_hd.jpg

 

根据以上的信息,我们就可以创建一个基于条件判断的playbook文件test.yml

 

v2-310edd0c88f453a4ae3d437100bc6de0_hd.jpg

 

 

v2-d556ce65d24c567a6538063ec19e6cfb_hd.jpg

 

为了演示效果,实验前把CentOS6的nginx先卸载掉,此处70的报错只是因为ssh缺少那边没有提供公钥文件,此处就不再演示

 

v2-e393a55f43564a0097d40db4c2f9e88e_hd.jpg

 

检查playbook语法有没有问题

 

v2-f6c29b286d11f3211010e50de9e54a50_hd.jpg

 

测试运行,没有报错,可以看出当执行service nginx start命令时候,只有CentOS6的主机执行了命令,不过开始那里提示有skipping信息是为什么?CentOS7开始也提示有skipping信息?但是后面确实是执行成功了,下面正式运行该playbook看看效果。

 

v2-53c29502af94c1cbb600cd4ae5ddc31f_hd.jpg

 

正式运行,似乎没有报什么错误

 

v2-0a79ee8524c2d5c6cc24dd53b0113f1d_hd.jpg

 

看看服务是否已经开启,此处可见80端口已经开发,应该是没有问题的,此处就不浏览主页做测试了

 

 

循环 :迭代,需要重复执行的任务;

对迭代项的引用,固定变量名为"item”,使用with_item属性给定要迭代的元素;

元素:列表

字符串

字典

基于字符串列表给出元素示例:

示例:基于列表的方式安装多个安装包

 

v2-f94b84a9f92ea7e301509264df995f0d_hd.jpg

 

 

v2-e7a52f5d932b7f4ef91447ea6056d443_hd.jpg

 

检查语法

 

v2-aca0361e4acc0c2b1808f2e3aac1214a_hd.jpg

 

测试运行,没有报错(这里就以69和70两台不同的版本的CentOS来做测试)

 

v2-626c879f183d8c5de11d6f558bda0b2b_hd.jpg

 

正式运行,69的机器报错了,看了一下原因,是下载php-mbstring的时候出错了,此处原因应该是虚拟挂载CentOS6.8的cd1导致的,挂载cd2应该就解决此问题,不过部分安装包应该是在cd1里面,所以小伙伴们最好找一个安全包都全的yum仓库

 

v2-f799d6bd99ff43ead3b0f54d9a09a931_hd.jpg

 

重新配置好yum仓库,并且把先前安装的先卸载掉,以配置实验

 

v2-b03a59970007d61976da771b1aa42d1f_hd.jpg

 

此处可以看出来,由于69主机刚报错了一个,所以所有的包都没有安装,7上面倒是都已经安装过了

 

v2-e2dfc717e23ca701bf84e9b5ee95cd3e_hd.jpg

 

重新运行脚本,没有报错了

 

v2-e20677df563f038fbc13eeed0c79ebcf_hd.jpg

 

验证,发现已经安装上了,此处就不再看其他安装包的安装情况了,应该没有大问题

 

v2-47e02e2942f46d9d6dd98584dfd51d7c_hd.jpg

 

基于字典列表给元素示例:

示例:创建指定的用户并属于指定的组

 

v2-b826b1dd56ce9c3cd0f4653641b9f3f2_hd.jpg

 

 

v2-01fc3fc7ce27405bcdcebec431c46289_hd.jpg

 

检查语法

 

v2-1c45814fbe55673be172210c973c42b4_hd.jpg

 

测试运行,没有提示有任何变化?

 

v2-deaee6eba755ec0bfbaa23d3da583c54_hd.jpg

 

正式运行,可以看见创建了对应的用户和组

 

v2-f5a9d71884c3937a50524fa4f416f817_hd.jpg

 

验证,符合我们预期

 

v2-abe5fb5f4db230ec9c7d924f91f95cf9_hd.jpg

 

角色:roles

以特定的层级目录结构进行组织的tasks、variables、handlers、templates、files等;

role_name/

files/:存储由copy或script等模块调用的文件;

tasks/:此目录中至少应该有一个名为main.yml的文件,用于定义各task;其它的文件需要由main.yml进行“包含”调用;

handlers/:此目录中至少应该有一个名为main.yml的文件,用于定义各handler;其它的文件需要由main.yml进行“包含”调用;

vars/:此目录中至少应该有一个名为main.yml的文件,用于定义各variable;其它的文件需要由main.yml进行“包含”调用;

templates/:存储由template模块调用的模板文本;

meta/:此目录中至少应该有一个名为main.yml的文件,定义当前角色的特殊设定及其依赖关系;其它的文件需要由main.yml进行“包含”调用;

default/:此目录中至少应该有一个名为main.yml的文件,用于设定默认变量;

在playbook中调用角色的方法:

- hosts: HOSTS

remote_user: USERNAME

roles:

- ROLE1

- ROLE2

- { role: ROLE3, VARIABLE: VALUE, ...}

- { role: ROLE4, when: CONDITION }

示例:创建对应的服务目录下面的模版

 

v2-9344a0da3fa29edb4fb7f4ac1c1fd0a9_hd.jpg

 

首先创建对应的目录

 

v2-aba8ea7d5940e9f6c2512970371cb1a7_hd.jpg

 

确认一下目录是否正确

 

v2-f02f8d3a2558b1d0409f2990807f5064_hd.jpg

 

首先准备一个安装包,放到nginx/file/目录下面

 

v2-c48daa6a180e69bbac8c9141ad64afe3_hd.jpg

 

新建一个nginx的task模板

 

v2-ebc6ba0da6be554b4ce502efc7ae9075_hd.jpg

 

大家可以发现此处的模板跟之前的不一样,比如说,开头没有了定义主机、用户、和task等,此处的task会自行查找/etc/ansible/roles/nginx/task/main.yml的任务(此处文件本身也是在task目录下面)。再比如说,该处指定的copy命令的src=FILENAME也是相对路径,其绝对路径为/etc/ansible/roles/nginx/file/FILENAME。又比如说定义了notify但是这里并没有handlers,是因为此处定义了的notify的名字会自行去查看该目录下面即/etc/ansible/roles/nginx/handlers/main.yml里面的handlers。还有template那里,大家有没发现也是用的相对路径,此处绝对路径为在/etc/ansible/roles/nginx/template/nginx.conf.j2。所以大家清楚了吗?

 

v2-c084cee3708d8fa75802242c5218ad49_hd.jpg

 

接着是定义/etc/ansible/roles/nginx/handlers/main.yml

 

v2-5f635b27a69d9ecce9152d284f404740_hd.jpg

 

此处的文件就是用来承上面的notify里面为什么没有定义的handlers的原因,因为已经定义在../handlers/main.yml里面了。

 

v2-d8f8bf9e0f56e7e1db25684a814cd7a0_hd.jpg

 

复制nginx.conf文件到templates目录下面为nginx.conf.j2

 

v2-debdf6b568503d6fdee4d8cfbd0f9e0f_hd.jpg

 

编辑里面定义的cpu数量,之前是2,所以此处我们也可以利用算数表达式来控制cpu的数量,比如此处-1,到时候看到的cpu数量应该是为1。

 

v2-aa0c48bb31666ca2fa863425f16beae8_hd.jpg

 

再复制/etc/nginx/conf.d/default.conf到nginx/templates/default.conf.j2

 

v2-d744e51ac19ae536704bac6d67fdcade_hd.jpg

 

然后编辑此文件

 

v2-cfba0cae9449ecaae3c9cdb9c46932ac_hd.jpg

 

编辑原来的端口号为一个变量值ngxport

 

v2-b45c8c1e7ae3989eb7542bad90f3a321_hd.jpg

 

此时需要重新编辑task/main.yml文件

 

v2-98afe88df1f2b9887853967135c13fd6_hd.jpg

 

主要添加一下内容

 

v2-99325456e28f685ab9bc7866aec25925_hd.jpg

 

此时我们就可以定义变量文件了

 

v2-a920d0be60e416b6cb97a0896358b346_hd.jpg

 

比如说定义ngxport的变量为8888

 

v2-e50f2073ab6a2c0733848af092bb572c_hd.jpg

 

此时所有的元素暂时都足够了,meta和default的文件夹在此处暂时用不上,然后我们在/etc/ansible/目录下面创建一个nginx.yml的文件

 

v2-e9877ac55d8e876c548da145cbb044a2_hd.jpg

 

注意此处的roles里面的nginx要在/etc/ansible.cfg文件里面有对应的设定

 

编辑查看ansible.cfg文件

 

v2-ae1b7ba94de9c2debcb0b259aeb39e99_hd.jpg

 

可以看见系统默认的roles路径也是在此处,所以我们去掉#号来启用它

 

v2-635d99341ce05bfd8bb299fe1b48f5a1_hd.jpg

修改成如下

 

v2-e4f43dd4e888b57b8fea58fe6375a679_hd.jpg

 

以上都准备好了以后,检查一下nginx.yml语法,暂时并没有报错

 

v2-97d504e6b7e42fcdef5bddd41cbd4592_hd.jpg

 

然后测试运行,可以看见此处报错了,看了一下报错的原因,是因为找不到/tmp/nginx安装包,因为只是测试运行,并没有传送安装包到目标主机上面,所以此处报错是正常的可以不予理会。

 

v2-50ff6c555b3904bf1a9ba53aa8622a21_hd.jpg

 

下面正式运行该脚本,此处报错了,原因看了一下,nginx安装包是el7版本的,在centos6上面并不能安装。以及handlers出问题了。

 

v2-e558d2f3fed10fb6135dc15251159f70_hd.jpg

 

此处修改一下tasks/main.yml,以下红色内容为修改部分,意思就是,CentOS7系统从远程复制的安装包安装,CentOS6则直接从yum仓库源安装,6和7的nginx的配置文件应该是一样的,暂时先这么操作实验看看结果,并且先把CentOS7系统的nginx安装包删除掉,以重新演示效果。notify处的语法错误,此处补上。

 

v2-9cd5bf72dc6489d06856cb45fa48c7ea_hd.jpg

 

修改完以上的内容,重新测试运行

可以看出来此处还是有报错内容,

第一个报错内容为找不到安装包,此处是正常的,因为安装包还是传过去(上一次运行的时候传送过去的安装包我已经删掉了,所以此处需要重传)

第二个报错内容为找不到nginx服务,此处也是正常的,因为nginx安装包还没有安装

 

v2-9b7c9c4ba93f8a012b36ecafbeabc45e_hd.jpg

 

正常重新运行一下nginx.yml脚本看看,发现已经没有报错的地方了

 

v2-d1cb82a45f32ded4b3c6b2ca0a3f49e4_hd.jpg

 

验证结果,发现8888端口已经打开

 

v2-8691f33be80000cb1897f3b93374cb15_hd.jpg

 

cpu数量的设置也跟我们之前配置的是一样的,实验到此结束

 

v2-542a57da3f9ba034846a897990c479de_hd.jpg

 

示例 2: 根据以上内容,修改端口号

当我们写好模板以后,需要修改端口号,也是非常容易的,而且我们也可以通过在nginx.yml上面通过roles传递变量

 

v2-2bdc0db1e902889060e10e2c3d2202c8_hd.jpg

 

例如像以下这样子操作

 

v2-a4c3af32d0cd6feeaf418893abb1c890_hd.jpg

 

测试运行一下看看有没有错误,可以看见在复制配置文件和重启服务那里有了变化,这符合我们预期

 

v2-1faa8c7ee037d079fe8064bcde72fecb_hd.jpg

 

正式运行一下看看,能正常运行

 

v2-acb579e1a87fc6e97589c25946e169f9_hd.jpg

 

验证端口号是否修改成功,看到8080端口,表示操作没有问题

 

v2-c95fbcad2f3a6eb9e5d927424c4d951c_hd.jpg

 

以上是运行成功了,但是细心的同学会发现,这样所有程序都跑一遍也麻烦,所以我们可以用之前了解到的标签来执行特定的操作即可,也可以直接传递相应的变量。

 

v2-cb2580ee198e66c9cfe8b853e175a48f_hd.jpg

 

直接传递参数测试运行,好像没有问题

 

v2-b2428f044683f8eccd94d3c20773287a_hd.jpg

 

正式运行

 

v2-299d82024155a99f82a35517211e15bf_hd.jpg

 

查看端口号是否正确,此处可以看见是我们定义的8099端口,测试成功

 

v2-844b9ef5b7435cdb6949aa1229f0c5a7_hd.jpg

 

示例:实现 httpd 不同主机不同的端口号

首先先把定义的端口号先屏蔽掉

 

v2-c931307506c23a37cfdff27b5765b547_hd.jpg

 

 

v2-8fef3802b7278d7c4bb4026314b68509_hd.jpg

 

编辑/etc/ansible/hosts文件

 

v2-4365f0c5e748aa9efd1ff2703fd5dd83_hd.jpg

 

定义对应的端口号,然后测试

 

v2-9323b46a47c002b3bd3f35a29c90aead_hd.jpg

 

然后记得把nginx.yml文件里面也改回来

 

v2-3481907e75e6cb539851d01c228748a4_hd.jpg

 

这里直接运行就不先做测试了,不过一般同学们还是做好测试工作比较好,本人比较懒O.O

 

v2-86234d43d6754fc9dbd9a05e3c20ab75_hd.jpg

 

验证端口号,也符合我们预期

 

v2-fd40c4e8e3723bb2262c5a38cca142d1_hd.jpg

 

示例:在同一个 yml 配置文件里面运行两个服务模板程序

这里以memcached为例,首先复制memcached的配置文件到对应的templates目录下来为.j2的文件

 

v2-78fcec6ae4399a93dbf5f2527d655b0e_hd.jpg

 

memcached服务是依靠设置内存参数来定义的,所以我们得首先用ansible来确认系统的内存变量参数值是什么,并且通过以下图可以看见两个系统参数都是一致的。

 

v2-99cf5dc2bffa96c44048fbd3dc86047c_hd.jpg

 

编辑memcached.j2文件

 

v2-6a2095c5220fdb2cb4d50363742bbbaf_hd.jpg

 

定义变量参数

 

v2-d8c3109ced097a5fe08b1adc2e7a718a_hd.jpg

改成

v2-32c0e1d419a8316c33834418f503af8a_hd.jpg

 

开始定义memcached的任务文件

 

v2-aad5286851a9c522538a1948666b7afe_hd.jpg

 

 

v2-154f2b09a9622ff12a3552a8f0539ebb_hd.jpg

 

定义handlers文件

 

v2-8c59321623fb2ca2506d90f833152d67_hd.jpg

 

 

v2-1dd710ebea39aac89f8c34eef1ae9f2b_hd.jpg

 

把memcached定义在ngnix.yml文件一同运行

 

v2-b89a90bd16986ffd2ca9ce5496bdc780_hd.jpg

 

 

v2-53e318030a732cd615a2f7719e4679a5_hd.jpg

 

测试运行,只是安装包还没有安装,提示的错误都问题不大,是正常的

 

v2-41ae5d743821c1bb27ce4a3fcdd75420_hd.jpg

 

正式运行

 

v2-482fe8d7bb67a0b17fe41b207e91f2ac_hd.jpg

 

验证服务是否开启,且是否设置好预期可用内存

可以看见11211端口已经打开

 

v2-5f85644e100bc4fd7652096837208f03_hd.jpg

 

查看可用内存,原来的数值是970~980多,这里200多,符合除以4的预期效果

 

v2-417cdc9b27f7b08a39194c4ba0c9b466_hd.jpg

 

示例 : 根据不同的系统安装 mysql 包

首先定义一个tasks的模板

 

v2-0e962247329574ce8fe54bba4b67610a_hd.jpg

 

 

v2-6e0bceb56e1f6170f78e1ff1eb1ec28c_hd.jpg

 

定义一个yaml调用角色脚本

 

v2-6d8ddc1b946715920073501265d73da4_hd.jpg

 

 

v2-055551e5e0af692a1c809d6c184942dc_hd.jpg

 

设置hosts文件添加dbsrvs组

 

v2-d9062721372a8532d1d55f74d8d1412e_hd.jpg

 

 

v2-645c9c14e6e6c16133903672011ed5b6_hd.jpg

 

测试语法

 

v2-d3eaefa2d3dff839f4bcfc6d968e3919_hd.jpg

 

测试运行调用角色脚本db.yml,应该没有大问题

 

v2-ab6ffbd22c2d5675aeb216e8ecfdfaf9_hd.jpg

 

正式运行,没有报任何错误

 

v2-0189e3f7f584f7615be3ebe9cc3779a8_hd.jpg

 

验证服务是否已经开启

可以看见mysql和mariadb服务均已经开启

 

v2-61a7a1567a8b823493722c9bd8803434_hd.jpg

 

写在最后,关于ansible的能最多控制几台主机

此处是在配置文件里面定义的,默认是5台主机,如果把主机的控制的主机调大,估计也要相对应性能的主机当ansible服务器

 

v2-572e4e23c59aee79e27e66f6f32e8b8b_hd.jpg

 

 

v2-54ee51b8184d5546236eb6a752baffe5_hd.jpg

 

至此,本博文已经完结,下面总结几个小点:

1、 ansible的playbook.yml文件要求的格式比较严格,有时候少了几个空格,或者空格位置不妥当的时候,系统均默认此格式为错误,所以需要小心

2、 有时候输入错了ansibile不能识别的错误,用- - syntax-check 或者 �Ccheck 测试文件的时候并不会提示有任何提示,需要实际运行才能会报错。

3、 有一次写playbook.yml文件的时候,检查过是没有问题,但是测试一直出问题,后来把所有重写一遍就好,也可能是哪里错了自己没看见。

作者:~微风~
来源: http://weiweidefeng.blog.51cto.com/1957995/1895261

我们整理了一个学习知识库,可以了解看一下:

【超全整理】《Linux云计算从入门到精通》系列实战笔记全放送 | Linux运维部落​www.magedu.com

 

希望大家有所帮助,祝愿大家学有所成。获取更多技术知识点+v156 5219 9186,欢豆在线解答哦~

转载于:https://my.oschina.net/u/4154787/blog/3089719

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值