自动化运维工具Ansible使用介绍

本文主要内容均收集于网络上的博文资料,仅以此文作为学习总结。BTW,目前Ansible对python3的支持还不是很完备。

一、基础介绍

1、 简介

ansible是新出现的自动化运维工具,基于Python开发,集合了众多运维工具(puppet、cfengine、chef、func、fabric)的优点,实现了批量系统配置、批量程序部署、批量运行命令等功能。ansible是基于模块工作的,本身没有批量部署的能力。真正具有批量部署的是ansible所运行的模块,ansible只是提供一种框架。主要包括:
(1)、连接插件connection plugins:负责和被监控端实现通信;
(2)、host inventory:指定操作的主机,是一个配置文件里面定义监控的主机;
(3)、各种模块核心模块、command模块、自定义模块;
(4)、借助于插件完成记录日志邮件等功能;
(5)、playbook:剧本执行多个任务时,非必需可以让节点一次性运行多个任务。

2、总体架构

总体架构

3、特性

(1)、no agents:不需要在被管控主机上安装任何客户端;
(2)、no server:无服务器端,使用时直接运行命令即可;
(3)、modules in any languages:基于模块工作,可使用任意语言开发模块;
(4)、yaml,not code:使用yaml语言定制剧本playbook;
(5)、ssh by default:基于SSH工作;
(6)、strong multi-tier solution:可实现多级指挥。

4、优点

(1)、轻量级,无需在客户端安装agent,更新时,只需在操作机上进行一次更新即可;
(2)、批量任务执行可以写成脚本,而且不用分发到远程就可以执行;
(3)、使用python编写,维护更简单,ruby语法过于复杂;
(4)、支持sudo。

5、任务执行流程

任务执行流程

二、安装Ansible

  • 从github下载源码:
$ git clone https://github.com/ansible/ansible.git --recursive
$ cd ./ansible
  • 运行环境配置脚本:
$ source ./hacking/env-setup
  • 如果没有安装pip,请先安装对应于你的Python版本的pip:
$ sudo easy_install pip

以下的Python模块也需要安装:

$ sudo pip install paramiko PyYAML Jinja2 httplib2 six pycrypto

如果安装paramiko 的时候系统报openssl相关的错误,那么需要再安装以下文件:

$ sudo apt-get install -y libffi-dev libssl-dev python-dev

当更新ansible版本时,不只要更新git的源码树,也要更新git中指向Ansible自身模块的 “submodules” (不是同一种模块):

$ git pull --rebase
$ git submodule update --init --recursive

三、Ansible 配置

1、SSH免密钥登录设置

  • 使用ssh-keygen生成key-pair,然后将公钥配置在每台服务器.ssh/authorized_keys文件中:
# ssh-keygen -t rsa -P ''
  • 写入信任文件(将/root/.ssh/id_rsa.pub分发到其他服务器,并在所有服务器上执行如下指令):
# cat /root/.ssh/id_rsa.pub >> /root/.ssh/authorized_keys
# chmod 600 /root/.ssh/authorized_keys

如果需要配置公钥的服务器数量比较多的话,也可以通过使用ansible中的authorize_key模块来进行分布式部署。

2、Ansible配置

Ansible使用前通常需要配置/etc/anisble/路径下的hosts和ansible.cfg这个两个文件。要注意的是,编辑这两个文件需要root权限。

# mkdir -p /etc/ansible`
# vim /etc/ansible/ansible.cfg`
……
remote_port = 36000
private_key_file = /root/.ssh/id_rsa
……

Ansible 可同时操作属于一个组的多台主机,组和主机之间的关系通过 inventory 文件(INI 格式)配置,默认的文件路径为 /etc/ansible/hosts。一个系统可以属于不同的组,比如一台服务器可以同时属于 webserver组 和 dbserver组。

# vim /etc/ansible/hosts
[test]
10.139.13.80

/etc/ansible/hosts是配置分组的默认文件,还有一种方式可以通过定义环境变量ANSIBLE_HOSTS来改变配置文件的引用路径:

export ANSIBLE_HOSTS=~/test/qiuyi_ansible/test_hosts

四、Ansible ad-hoc命令

Ansible提供两种方式去完成任务,一种是 ad-hoc 命令,另一种写 Ansible playbook。前者可以解决一些简单的任务,后者解决较复杂的任务。ad-hoc只能做些一些轻量级的指令操作,而配置管理或部署这种事还是要借助 playbook 来完成,即使用 ‘/usr/bin/ansible-playbook’ 这个命令。

ad-hoc的命令格式:

ansible <host-pattern> [options]

ansible有许多模块,默认是 ‘command’,也就是命令模块,我们可以通过 -m 选项来指定不同的模块。

常用模块:

1. setup:用来查看远程主机的一些基本信息

2. ping:用来测试远程主机的运行状态

3. file:设置文件属性

相关选项如下:

  • force:需要在两种情况下强制创建软链接,一种是源文件不存在,但之后会建立的情况下;另一种是目标软链接已存在,需要先取消之前的软链,然后创建新的软链,有两个选项:yes|no

  • group:定义文件/目录的属组

  • mode:定义文件/目录的权限

  • owner:定义文件/目录的属主

  • path:必选项,定义文件/目录的路径

  • recurse:递归设置文件的属性,只对目录有效

  • src:被链接的源文件路径,只应用于state=link的情况

  • dest:被链接到的路径,只应用于state=link的情况

  • state

      state:
      directory:如果目录不存在,就创建目录
      file:即使文件不存在,也不会被创建
      link:创建软链接
      hard:创建硬链接
      touch:如果文件不存在,则会创建一个新的文件,如果文件或目录已存在,则更新其最后 修改时间
      absent:删除目录、文件或者取消链接文件
    

示例

  • 远程文件符号链接创建
# ansible test -m file -a "src=/etc/resolv.conf dest=/tmp/resolv.conf state=link"
  • 远程文件信息查看
# ansible test -m command -a "ls –al /tmp/resolv.conf"
  • 远程文件符号链接删除
# ansible test -m file -a "path=/tmp/resolv.conf state=absent"
  • 远程文件信息查看
# ansible test -m command -a "ls -al /tmp/resolv.conf"

4. copy:复制文件到远程主机

相关选项如下:

  • backup:在覆盖之前,将源文件备份,备份文件包含时间信息。有两个选项:yes|no
  • content:用于替代“src”,可以直接设定指定文件的值
  • dest:必选项。要将源文件复制到的远程主机的绝对路径,如果源文件是一个目录,那么该路径也必须是个目录
  • directory_mode:递归设定目录的权限,默认为系统默认权限
  • force:如果目标主机包含该文件,但内容不同,如果设置为yes,则强制覆盖,如果为no,则只有当目标主机的目标位置不存在该文件时,才复制。默认为yes
  • others:所有的file模块里的选项都可以在这里使用
  • src:被复制到远程主机的本地文件,可以是绝对路径,也可以是相对路径。如果路径是一个目录,它将递归复制。在这种情况下,如果路径使用“/”来结尾,则只复制目录里的内容,如果没有使用“/”来结尾,则包含目录在内的整个内容全部复制,类似于rsync。

示例:

  • 将本地文件“/etc/ansible/ansible.cfg”复制到远程服务器
# ansible test -m copy -a "src=/etc/ansible/ansible.cfg dest=/tmp/ansible.cfg owner=root group=root mode=0644"
  • 远程文件信息查看
# ansible test -m command -a "ls -al /tmp/ansible.cfg"

5. command:在远程主机上执行命令

相关选项如下:

  • creates:一个文件名,当该文件存在,则该命令不执行
  • free_form:要执行的linux指令
  • chdir:在执行指令之前,先切换到该目录
  • removes:一个文件名,当该文件不存在,则该选项不执行
  • executable:切换shell来执行指令,该执行路径必须是一个绝对路径

示例:

  • 如果远端服务器的~/qiuyi_test目录中已经存在1.txt文件的话,则不执行ls操作:
$ ansible test -m command  -a "chdir=~/qiuyi_test creates=~/qiuyi_test/1.txt ls"
10.139.13.80 | skipped
  • 如果远端服务器的"~/qiuyi_test"目录中已经不存在2.txt文件的话,
    则在"~/qiuyi_test"路径下执行ls操作:
$ ansible test -m command  -a "chdir=~/qiuyi_test creates=~/qiuyi_test/2.txt ls"
10.139.13.80 | success | rc=0 >>
1.txt
qiuyi.sh
ssh

注意,creates选项并不会创建文件。

6. shell:切换到某个shell执行指定的指令,参数与command相同

与command不同的是,此模块可以支持命令管道,同时还有另一个模块也具备此功能:raw

示例:

# ansible raleigh -m shell -a "echo $TERM"

使用 Ansible ad-hoc 命令行接口时(与使用 Playbooks 的情况相反),尤其注意 shell 引号的规则. 比如在上面的例子中,如果使用双引号”echo $TERM”,会求出TERM变量在当前系统的值,而我们实际希望的是把这个命令传递 到其它机器执行.

示例:

  • 先在本地创建一个SHELL脚本
$ vim /home/user_00/test/qiuyi_ansible/qiuyi_test.sh
#!/bin/sh
date +%F_%H:%M:%S
$ chmod +x /home/user_00/test/qiuyi_ansible/qiuyi_test.sh
  • 将创建的脚本文件分发到远程
$ ansible test -m copy -a "src= /home/user_00/test/qiuyi_ansible/qiuyi_test.sh dest=/home/user_00/qiuyi_test owner=user_00  mode=0755"
  • 远程执行
$ ansible test -m shell -a "~/qiuyi_test/qiuyi.sh"
10.139.13.80 | success | rc=0 >>
2017-03-22_12:27:18

7. 更多模块

其他常用模块,比如:service、cron、yum、synchronize就不一一例举,可以结合自身的系统环境进行测试。在使用中,如果想查看某个模块的信息,可以使ansible-doc命令。

  • service:系统服务管理
  • cron:计划任务管理
  • yum:yum软件包安装管理
  • synchronize:使用rsync同步文件
  • user:系统用户管理
  • group:系统用户组管理
# ansible-doc –l
add_host             add a host (and alternatively a group) to the ansible-playbo
apt                  Manages apt-packages                                        
apt_repository       Manages apt repositores                                     
assemble             Assembles a configuration file from fragments               
async_status         Obtain status of asynchronous task                          
authorized_key       Adds or removes an SSH authorized key                       
command              Executes a command on a remote node                         
copy                 Copies files to remote locations.                           
cron                 Manage crontab entries.                                     
debug                Print statements during execution                           
easy_install         Installs Python libraries                                   
ec2                  create an instance in ec2, return instanceid                
facter               Runs the discovery program `facter' on the remote system... 
fail                 Fail with custom message                                    
fetch                Fetches a file from remote nodes                            
file                 Sets attributes of files                                    
get_url              Downloads files from HTTP, HTTPS, or FTP to node            
git                  Deploy software (or files) from git checkouts               
group                Add or remove groups                                        
group_by             Create Ansible groups based on facts                        
ini_file             Tweak settings in INI files                                 
lineinfile           Ensure a particular line is in a file                       
mail                 Send an email                                               
mount                Control active and configured mount points                  
mysql_db             Add or remove MySQL databases from a remote host.           
mysql_user           Adds or removes a user from a MySQL database.               
nagios               Perform common tasks in Nagios related to downtime and notif
ohai                 Returns inventory data from `Ohai'                          
pause                Pause playbook execution                                    
ping                 Try to connect to host and return `pong' on success.        
pip                  Manages Python library dependencies.                        
postgresql_db        Add or remove PostgreSQL databases from a remote host.      
postgresql_user      Adds or removes a users (roles) from a PostgreSQL database..
raw                  Executes a low-down and dirty SSH command                   
script               Runs a local script on a remote node                        
seboolean            Toggles SELinux booleans.                                   
selinux              Change policy and state of SELinux                          
service              Manage services.                                            
setup                Gathers facts about remote hosts                            
shell                Execute commands in nodes.                                  
slurp                Slurps a file from remote nodes                             
subversion           Deploys a subversion repository.                            
supervisorctl        Manage the state of a program or group of programs running v
svr4pkg              Manage Solaris SVR4 packages                                
template             Templates a file out to a remote server.                    
user                 Manage user accounts                                        
virt                 Manages virtual machines supported by libvirt               
wait_for             Waits for a given port to become accessible on a server.... 
yum                  Manages packages with the `yum' package manager

五、 Palybooks

1、基础介绍

Playbooks与ad-hoc相比,是一种完全不同的运用 ansible 的方式。
简单来说,playbooks是一种简单的配置管理系统与多机器部署系统的基础。与现有的其他系统有不同之处,且非常适合于复杂应用的部署。Playbooks可用于声明配置,更强大的地方在于,在 playbooks中可以编排有序的执行过程,甚至于做到在多组机器间,来回有序的执行特别指定的步骤。并且可以同步或异步的发起任务。

Playbooks 采用的格式是YAML

---
- hosts: webservers
  vars:
    http_port: 80
    max_clients: 200
  remote_user: root
  tasks:
  - name: ensure apache is at the latest version
    yum: pkg=httpd state=latest
  - name: write the apache config file
    template: src=/srv/httpd.j2 dest=/etc/httpd.conf
    notify:
    - restart apache
  - name: ensure apache is running
    service: name=httpd state=started
  handlers:
    - name: restart apache
      service: name=httpd state=restarted
  • hosts 行的内容是一个或多个组或主机的 patterns,以逗号为分隔符;

  • remote_user 就是账户名;

  • vars为变量列表;

    在 action 行中可以使用变量。假设在 ‘vars’ 那里定义了一个变量 ‘vhost’ ,可以这样使用它:

tasks:
  - name: create a virtual host file for {{ vhost }}
    template: src=somefile.j2 dest=/etc/httpd/conf.d/{{ vhost }}
  • task的目标在于执行一个moudle;

    每一个 play 包含了一个 task 列表(任务列表)。一个 task 在其所对应的所有主机上(通过 host pattern 匹配的所有主机)执行完毕之后,下一个 task 才会执行。有一点需要明白的是,在一个 play 之中,所有 hosts 会获取相同的任务指令,这是 play 的一个目的所在,也就是将一组选出的 hosts 映射到 task。
    在运行playbook时(从上到下执行),如果一个host执行task失败,这个host将会从整个 playbook的rotation中移除。如果发生执行失败的情况,请修正playbook 中的错误,然后重新执行即可。每一个task必须有一个名称name,这样在运行playbook 时,从其输出的任务执行信息中可以很好的辨别出是属于哪一个task的。

    一个基本的tasks定义:

tasks:
  - name: make sure apache is running
    service: name=httpd state=running
  • Handlers: 在发生改变时执行的操作

    这里有一个例子,当一个文件的内容被改动时,重启两个 services,‘notify’ 下列出的即是 handlers:

- name: template configuration file
  template: src=template.j2 dest=/etc/foo.conf
  notify:
     - restart memcached
     - restart apache

Handlers 也是一些task的列表,通过名字来引用,它们和一般的 task 并没有什么区别。Handlers是由通知者进行 notify, 如果没有被 notify,handlers 不会执行。不管有多少个通知者进行了 notify,等到 play 中的所有 task 执行完成之后,handlers也只会被执行一次。handlers 会按照声明的顺序执行。

这里是一个 handlers 的示例:

handlers:
    - name: restart memcached
      service:  name=memcached state=restarted
    - name: restart apache
      service: name=apache state=restarted

Handlers 最佳的应用场景是用来重启服务,或者触发系统重启操作。除此以外很少用到了。

  • 执行一个playbook

以下示例是并行的运行playbook,并行的级别是10:

ansible-playbook playbook.yml -f 10

在执行一个 playbook 之前,想看看这个 playbook 的执行会影响到哪些 hosts,你可以这样做:

ansible-playbook playbook.yml --list-hosts

playbook是由一个或多个“play”组成的列表,可以让它们联同起来按事先编排的机制执行;所谓task无非是调用ansible的一个module,而在模块参数中可以使用变量;模块执行是幂等的,这意味着多次执行是安全的,因为其结果均一致。

幂等(idempotent、idempotence)是一个数学与计算机学概念,常见于抽象代数中。
在编程中,一个幂等操作的特点是其任意多次执行所产生的影响均与一次执行的影响相同。幂等函数,或幂等方法,是指可以使用相同参数重复执行,并能获得相同结果的函数。这些函数不会影响系统状态,也不用担心重复执行会对系统造成改变。

2、Roles和include语句

假如你希望在多个play或者多个playbook中重用同一个task列表,你可以使用include 做到这一点。 当我们希望为系统定义一个角色时,使用include去包含task列表是一种很好用的方法。需要记住的是,一个play所要达成的目标是将一组系统映射为多个角色。

一个 task include file 由一个普通的 task 列表所组成,像这样:

---
# possibly saved as tasks/foo.yml

- name: placeholder foo
  command: /bin/foo

- name: placeholder bar
  command: /bin/bar

Include 指令看起来像下面这样,在一个 playbook 中,Include 指令可以跟普通的 task 混合在一起使用:

tasks:

  - include: tasks/foo.yml

Roles 基于一个已知的文件结构,去自动的加载某些 vars_files,tasks 以及 handlers。基于 roles 对内容进行分组,使得我们可以容易地与其他用户分享 roles 。

一个项目的结构如下:

site.yml
webservers.yml
fooservers.yml
roles/
   common/
     files/
     templates/
     tasks/
     handlers/
     vars/
     defaults/
     meta/
   webservers/
     files/
     templates/
     tasks/
     handlers/
     vars/
     defaults/
     meta/

一个 playbook 如下:

---
- hosts: webservers
  roles:
     - common
     - webservers

这个 playbook 为一个角色 ‘x’ 指定了如下的行为:

  • 如果 roles/x/tasks/main.yml 存在, 其中列出的 tasks 将被添加到 play 中
  • 如果 roles/x/handlers/main.yml 存在, 其中列出的 handlers 将被添加到 play 中
  • 如果 roles/x/vars/main.yml 存在, 其中列出的 variables 将被添加到 play 中
  • 如果 roles/x/meta/main.yml 存在, 其中列出的 “角色依赖” 将被添加到 roles 列表中 (1.3 and later)
  • 所有 copy tasks 可以引用 roles/x/files/ 中的文件,不需要指明文件的路径。
  • 所有 script tasks 可以引用 roles/x/files/ 中的脚本,不需要指明文件的路径。
  • 所有 template tasks 可以引用 roles/x/templates/ 中的文件,不需要指明文件的路径。
  • 所有 include tasks 可以引用 roles/x/tasks/ 中的文件,不需要指明文件的路径。

3、小结

本节只对Playbooks的基础内容进行了简单的介绍;而余下的如Roles依赖、条件控制、循环操作、异步处理等相关内容可以参考Ansible中文权威指南进行学习,本文将不再深入介绍。

六、参考文献

Reference:
[1] http://www.ansible.com.cn/docs/
[2] http://sofar.blog.51cto.com/353572/1579894
[3] http://baike.baidu.com/item/ansible
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,219评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,363评论 1 293
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,933评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,020评论 0 206
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,400评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,640评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,896评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,597评论 0 199
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,327评论 1 244
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,581评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,072评论 1 261
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,399评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,054评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,083评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,849评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,672评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,585评论 2 270

推荐阅读更多精彩内容