Python自动化运维和部署项目工具Fabric使用实例


Posted in Python onSeptember 18, 2016

Fabric 是使用 Python 开发的一个自动化运维和部署项目的一个好工具,可以通过 SSH 的方式与远程服务器进行自动化交互,例如将本地文件传到服务器,在服务器上执行shell 命令。

下面给出一个自动化部署 Django 项目的例子

# -*- coding: utf-8 -*-
# 文件名要保存为 fabfile.py

from __future__ import unicode_literals
from fabric.api import *

# 登录用户和主机名:
env.user = 'root'
# 如果没有设置,在需要登录的时候,fabric 会提示输入
env.password = 'youpassword'
# 如果有多个主机,fabric会自动依次部署
env.hosts = ['www.example.com']

TAR_FILE_NAME = 'deploy.tar.gz'

def pack():
  """
  定义一个pack任务, 打一个tar包
  :return:
  """
  tar_files = ['*.py', 'static/*', 'templates/*', 'vue_app/', '*/*.py', 'requirements.txt']
  exclude_files = ['fabfile.py', 'deploy/*', '*.tar.gz', '.DS_Store', '*/.DS_Store',
           '*/.*.py', '__pycache__/*']
  exclude_files = ['--exclude=\'%s\'' % t for t in exclude_files]
  local('rm -f %s' % TAR_FILE_NAME)
 
  local('tar -czvf %s %s %s' % (TAR_FILE_NAME, ' '.join(exclude_files), ' '.join(tar_files)))
  print('在当前目录创建一个打包文件: %s' % TAR_FILE_NAME)


def deploy():
  """
  定义一个部署任务
  :return:
  """
  # 先进行打包
  pack()

  # 远程服务器的临时文件
  remote_tmp_tar = '/tmp/%s' % TAR_FILE_NAME
  run('rm -f %s' % remote_tmp_tar)
  # 上传tar文件至远程服务器, local_path, remote_path
  put(TAR_FILE_NAME, remote_tmp_tar)
  # 解压
  remote_dist_base_dir = '/home/python/django_app'
  # 如果不存在, 则创建文件夹
  run('mkdir -p %s' % remote_dist_dir)

 # cd 命令将远程主机的工作目录切换到指定目录 
  with cd(remote_dist_dir):
    print('解压文件到到目录: %s' % remote_dist_dir)
    run('tar -xzvf %s' % remote_tmp_tar)
    print('安装 requirements.txt 中的依赖包')
    # 我使用的是 python3 来开发
    run('pip3 install -r requirements.txt')
    remote_settings_file = '%s/django_app/settings.py' % remote_dist_dir
    settings_file = 'deploy/settings.py' % name
    print('上传 settings.py 文件 %s' % settings_file)
    put(settings_file, remote_settings_file)

    nginx_file = 'deploy/django_app.conf'
    remote_nginx_file = '/etc/nginx/conf.d/django_app.conf'
    print('上传 nginx 配置文件 %s' % nginx_file)
    put(nginx_file, remote_nginx_file)
 
 # 在当前目录的子目录 deploy 中的 supervisor 配置文件上传至服务器
  supervisor_file = 'deploy/django_app.ini'
  remote_supervisor_file = '/etc/supervisord.d/django_app.ini'
  print('上传 supervisor 配置文件 %s' % supervisor_file)
  put(supervisor_file, remote_supervisor_file)
 
 # 重新加载 nginx 的配置文件
  run('nginx -s reload')
  run('nginx -t')
  # 删除本地的打包文件
  local('rm -f %s' % TAR_FILE_NAME)
  # 载入最新的配置文件,停止原有进程并按新的配置启动所有进程
  run('supervisorctl reload')
  # 执行 restart all,start 或者 stop fabric 都会提示错误,然后中止运行
  # 但是服务器上查看日志,supervisor 有重启
  # run('supervisorctl restart all')

执行 pack 任务

fab pack

执行 deploy 任务

fab deploy

再给大家分享一个使用Fabric进行代码的自动化部署

#coding=utf-8
from fabric.api import local, abort, settings, env, cd, run
from fabric.colors import *
from fabric.contrib.console import confirm

env.hosts = ["root@115.28.×××××"]
env.password = "×××××"


def get_git_status():
  git_status_result = local("git status", capture=True)
  if "无文件要提交,干净的工作区" not in git_status_result:
    print red("****当前分支还有文件没有提交")
    print git_status_result
    abort("****已经终止")


def local_unit_test():
  with settings(warn_only=True):
    test_result = local("python manage.py test")
    if test_result.failed:
      print test_result
      if not confirm(red("****单元测试失败,是否继续?")):
        abort("****已经终止")


def server_unit_test():
  with settings(warn_only=True):
    test_result = run("python manage.py test")
    if test_result.failed:
      print test_result
      if not confirm(red("****单元测试失败,是否继续?")):
        abort("****已经终止")


def upload_code():
  local("git push origin dev")
  print green("****代码上传成功")


def deploy_at_server():
  print green("****ssh到服务器进行下列操作")
  with cd("/var/www/××××××"):
    #print run("pwd")
    print green("****将在远程仓库下载代码")
    run("git checkout dev")
    get_git_status()
    run("git pull origin dev")
    print green("****将在服务器上运行单元测试")
    server_unit_test()
    run("service apache2 restart", pty=False)
    print green("****重启apache2成功")
    print green("********代码部署成功********")


def deploy():
  get_git_status()
  local("git checkout dev", capture=False)
  print green("****切换到dev分支")
  get_git_status()
  print green("****将开始运行单元测试")
  local_unit_test()
  print green("****单元测试完成,开始上传代码")
  upload_code()
  deploy_at_server()

fabric可以将自动化部署或者多机操作的命令固化到一个脚本里,从而减少手动的操作。上面是今天第一次接触这东西后写的,确实很实用。运行fab deploy就行了。

主要逻辑就是将本地的dev分支跑单元测试,然后提交到服务器,ssh登陆到服务器,然后pull下来,再跑单元测试,然后重启apache2。第一次写,可能比较简单,将持续改进。

Python 相关文章推荐
跟老齐学Python之不要红头文件(1)
Sep 28 Python
Python调用C语言开发的共享库方法实例
Mar 18 Python
PyQt5利用QPainter绘制各种图形的实例
Oct 19 Python
Python3.6使用tesseract-ocr的正确方法
Oct 17 Python
python数据处理 根据颜色对图片进行分类的方法
Dec 08 Python
matplotlib实现热成像图colorbar和极坐标图的方法
Dec 13 Python
pytorch:实现简单的GAN示例(MNIST数据集)
Jan 10 Python
基于PyTorch的permute和reshape/view的区别介绍
Jun 18 Python
keras实现VGG16方式(预测一张图片)
Jul 07 Python
opencv 图像腐蚀和图像膨胀的实现
Jul 07 Python
Python编写memcached启动脚本代码实例
Aug 14 Python
详解Python中import机制
Sep 11 Python
基于Python 的进程管理工具supervisor使用指南
Sep 18 #Python
打包发布Python模块的方法详解
Sep 18 #Python
在python的类中动态添加属性与生成对象
Sep 17 #Python
Python中字符串的处理技巧分享
Sep 17 #Python
Python中对象迭代与反迭代的技巧总结
Sep 17 #Python
发布你的Python模块详解
Sep 15 #Python
Python selenium 三种等待方式解读
Sep 15 #Python
You might like
php $_ENV为空的原因分析
2009/06/01 PHP
在VS2008中编译MYSQL5.1.48的方法
2010/07/03 PHP
smarty内置函数section的用法
2015/01/22 PHP
laravel 框架实现无限级分类的方法示例
2019/10/31 PHP
JavaScript DOM学习第八章 表单错误提示
2010/02/19 Javascript
基于JQuery的简单实现折叠菜单代码
2010/09/15 Javascript
JQuery魔力之$("tagName")与selector
2012/03/05 Javascript
Jquery封装tab自动切换效果的具体实现
2013/07/13 Javascript
js获取php变量的实现代码
2013/08/10 Javascript
实用的Jquery选项卡TAB示例代码
2013/08/28 Javascript
告诉你什么是javascript的回调函数
2014/09/04 Javascript
javascript使用smipleChart实现简单图表
2015/01/02 Javascript
JavaScript判断undefined类型的正确方法
2015/06/30 Javascript
JavaScript编程中的Promise使用大全
2015/07/28 Javascript
js判断浏览器类型及设备(移动页面开发)
2015/07/30 Javascript
jquery判断input值不为空的方法
2016/06/05 Javascript
node.js+captchapng+jsonwebtoken实现登录验证示例
2017/08/17 Javascript
web前端vue之CSS过渡效果示例
2018/01/10 Javascript
jQuery实现DIV响应鼠标滑过由下向上展开效果示例【测试可用】
2018/04/26 jQuery
详解React 的几种条件渲染以及选择
2018/10/23 Javascript
利用JS如何获取form表单数据
2019/12/19 Javascript
简单的Python抓taobao图片爬虫
2014/10/26 Python
numpy 计算两个数组重复程度的方法
2018/11/07 Python
面向对象学习之pygame坦克大战
2019/09/11 Python
在pytorch 中计算精度、回归率、F1 score等指标的实例
2020/01/18 Python
python+opencv边缘提取与各函数参数解析
2020/03/09 Python
完美解决pyinstaller打包报错找不到依赖pypiwin32或pywin32-ctypes的错误
2020/04/01 Python
python lambda的使用详解
2021/02/26 Python
HTML5实现锚点时请使用id取代name
2013/09/06 HTML / CSS
char型变量中能不能存贮一个中文汉字
2015/07/08 面试题
新闻发布会主持词
2014/03/28 职场文书
关于读书的演讲稿800字
2014/08/27 职场文书
夫妻分居协议书范文
2014/11/26 职场文书
关于学习的决心书
2015/02/05 职场文书
redis sentinel监控高可用集群实现的配置步骤
2022/04/01 Redis
Mysql排查分析慢sql之explain实战案例
2022/04/19 MySQL