fabfile 的编写
fab 命令是结合我们编写的 fabfile.py(其他文件名须添加 -f filename 引用)来搭配使用的,部分命令行参数可以通过相应的方法来代替,使之更加灵活,例如 "-H 192.168.1.21,192.168.1.22",我们可以通过定义 env.hosts 来实现,如 "env.hosts =['192.168.1. 21','192.168.1.22']"。fabfile 的主体由多个自定义的任务函数组成,不同任务函数实现不同的操作逻辑,下面详细介绍。
全局属性设定
evn 对象的作用是定义 fabfile 的全局设定,支持多个属性,包括目标主机、用户、密码、角色等,各属性说明如下:
-
env.host,定义目标主机,可以用 IP 或主机名表示,以 Python 的列表形式定义,如 env.hosts=['192.168.1.21','192.168.1.22']。
-
env.exclude_hosts,排除指定主机,如 env.exclude_hosts=['192.168.1.22']。
-
env.user,定义用户名,如 env.user="root"。
-
env.port,定义目标主机端口,默认为 22,如 env.port="22"。
-
env.password,定义密码,如 env.password='KSJ3548t7d'。
-
env.passwords,与 password 功能一样,区别在于不同主机不同密码的应用场景,需要注意的是,配置 passwords 时需配置用户、主机、端口等信息,如:
env.passwords = { 'root@192.168.1.21:22': 'SJk348ygd', 'root@192.168.1.22:22': 'KSh458j4f', 'root@192.168.1.23:22': 'KSdu43598' } -
env.gateway,定义网关(中转、堡垒机)IP,如 env.gateway = '192.168.1.23'。
-
env.deploy_release_dir,自定义全局变量,格式:env.+“变量名称”,如 env.deploy_release_dir、env.age、env.sex 等。
-
env.roledefs,定义角色分组,比如 web 组与 db 组主机区分开来,定义如下:
env.roledefs = {
'webservers': ['192.168.1.21', '192.168.1.22', '192.168.1.23', '192.168.1.24'],
'dbservers': ['192.168.1.25', '192.168.1.26']
}
引用时使用 Python 修饰符的形式进行,角色修饰符下面的任务函数为其作用域,下面来看一个示例:
@roles('webservers')
def webtask():
run('/etc/init.d/nginx start')
@roles('dbservers')
def dbtask():
run('/etc/init.d/mysql start')
@roles ('webservers', 'dbservers')
def pubclitask():
run('uptime')
def deploy():
execute(webtask)
execute(dbtask)
execute(pubclitask)
在命令行执行 #fab deploy 就可以实现不同角色执行不同的任务函数了。
常用API
Fabric 提供了一组简单但功能强大的 fabric.api 命令集,简单地调用这些 API 就能完成大部分应用场景需求。Fabric 支持常用的方法及说明如下:
-
local,执行本地命令,如:local('uname -s');
-
lcd,切换本地目录,如:lcd('/home');
-
cd,切换远程目录,如:cd('/data/logs');
-
run,执行远程命令,如:run('free -m');
-
sudo,sudo 方式执行远程命令,如:sudo('/etc/init.d/httpd start');
-
put,上传本地文件到远程主机,如:put('/home/user.info','/data/user.info');
-
get,从远程主机下载文件到本地,如:get('/data/user.info', '/home/root.info');
-
prompt,获得用户输入信息,如:prompt('please input user password:');
-
confirm,获得提示信息确认,如:confirm("Tests failed.Continue[Y/N]?");
-
reboot,重启远程主机,如:reboot();
-
@task,函数修饰符,标识的函数为 fab 可调用的,非标记对 fab 不可见,纯业务逻辑;
-
@runs_once,函数修饰符,标识的函数只会执行一次,不受多台主机影响。
下面结合一些示例来帮助大家理解以上常用的 API。
示例1:查看本地与远程主机信息
本示例调用 local() 方法执行本地(主控端)命令,添加 "@runs_once" 修饰符保证该任务函数只执行一次。调用 run() 方法执行远程命令。详细源码如下:
#!/usr/bin/env python
from fabric.api import *
env.user = 'root'
env.hosts = ['192.168.1.21', '192.168.1.22']
env.password = 'LKs934jh3'
@runs_once # 查看本地系统信息,当有多台主机时只运行一次
def local_task(): # 本地任务函数
local("uname -a")
def remote_task():
with cd("/data/logs"): # “with”的作用是让后面的表达式的语句继承当前状态,实现
run("ls -l") # “cd /data/logs && ls -l”的效果
通过 fab 命令分别调用 local_task 任务函数运行结果如图7-2所示。
结果中显示了 "[192.168.1.21] Executing task 'local_task'",但事实上并非在主机 192.168.1.21 上执行任务,而是返回 Fabric 主机本地 "uname -a" 的执行结果。
调用 remote_task 任务函数的执行结果如图7-3所示。
示例2:动态获取远程目录列表
本示例使用 @task 修饰符标志入口函数 go() 对外部可见,配合 @runs_once 修饰符接收用户输入,最后调用 worktask() 任务函数实现远程命令执行,详细源码如下:
#!/usr/bin/env python
from fabric.api import *
env.user = 'root'
env.hosts = ['192.168.1.21', '192.168.1.22']
env.password = 'LKs934jh3'
@runs_once # 主机遍历过程中,只有第一台触发此函数
def input_raw():
return prompt("please input directory name:", default="/home")
def worktask(dirname):
run("ls -l " + dirname)
@task # 限定只有go函数对fab命令可见
def go():
getdirname = input_raw()
worktask(getdirname)
该示例实现了一个动态输入远程目录名称,再获取目录列表的功能,由于我们只要求输入一次,再显示所有主机上该目录的列表信息,调用了一个子函数 input_raw() 同时配置 @runs_once 修饰符来达到此目的。执行结果如图7-4所示。
执行结果如图7-4所示。
示例3:网关模式文件上传与执行
本示例通过 Fabric 的 evn 对象定义网关模式,即俗称的中转、堡垒机环境。定义格式为 "env.gateway='192.168.1.23'",其中 IP"192.168.1.23" 为堡垒机 IP,再结合任务函数实现目标主机文件上传与执行的操作,详细源码如下:
#!/usr/bin/env python
from fabric.api import *
from fabric.context_managers import *
from fabric.contrib.console import confirm
env.user = 'root'
env.gateway = '192.168.1.23' # 定义堡垒机IP,作为文件上传、执行的中转设备
env.hosts = ['192.168.1.21', '192.168.1.22']
# 假如所有主机密码都不一样,可以通过env.passwords字典变量一一指定
env.passwords = {
'root@192.168.1.21:22': 'LKs934jh3',
'root@192.168.1.22:22': 'LKs934jh3',
'root@192.168.1.23:22': 'UI7384hg6' # 堡垒机账号信息
}
lpackpath = "/home/install/lnmp0.9.tar.gz" # 本地安装包路径
rpackpath = "/tmp/install" # 远程安装包路径
@task
def put_task():
run("mkdir -p /tmp/install")
with settings(warn_only=True):
result = put(lpackpath, rpackpath) # 上传安装包
if result.failed and not confirm("put file failed, Continue[Y/N]?"):
abort("Aborting file put task!")
@task
def run_task(): # 执行远程命令,安装lnmp环境
with cd("/tmp/install"):
run("tar -zxvf lnmp0.9.tar.gz")
with cd("lnmp0.9/"): # 使用with继续继承/tmp/install目录位置状态
run("./centos.sh")
@task
def go(): # 上传、安装组合
put_task()
run_task()
示例通过简单的配置 env.gateway='192.168.1.23' 就可以轻松实现堡垒机环境的文件上传及执行,相比 paramiko 的实现方法简洁了很多,编写的任务函数完全不用考虑堡垒机环境,配置 env.gateway 即可。