python使用paramiko自动化部署linux程序
使用paramiko模块,比os模块和command模块更加的兼容各种环境。后面两个只能在服务器本机 执行,此模块写得python文件无论是在本地还是服务器本身上运行,都能兼容。
paramiko模块的用法如下。
"""自动部署,可以在本地运行执行linux命令,也可以在远程服务器链接自身运行""" import paramiko
from app.utils.utils_ydf import mixins, unittest_util class HotelFaresDeploy(mixins.LoggerMixin):
def __init__(self, project_env='ydf', ip='115.95.89.xx', port=22, user_name="webuser", user_password='xxxxxxx'):
"""
:param project_env: 项目环境。不同的环境有不同的文件夹
:param ip: linux服务器ip
:param port: 端口
:param user_name: 用户名
:param user_password: 密码
"""
hotel_fares_dir = None
if project_env == 'ydf':
hotel_fares_dir = '/home/xxx/hotelf/'
elif project_env == 'test':
hotel_fares_dir = '/hotel/hotel_fares'
if project_env not in ('ydf', 'test'):
raise Exception('设置的项目环境错误')
self.hotel_fares_dir = hotel_fares_dir
ssh = paramiko.SSHClient()
ssh.set_missing_host_key_policy(paramiko.AutoAddPolicy())
ssh.connect(ip, port, user_name, user_password)
self.ssh = ssh
self.ssh_session = self.ssh.get_transport().open_session()
self.cmd0 = 'cd {} ; export PYTHONPATH=$PYTHONPATH:{};'.format(self.hotel_fares_dir, self.hotel_fares_dir) # 先进入项目所在文件夹和添加环境变量,这样才能保证后面的程序能顺利执行,linux,运行多条命令用;
self.logger.debug('cmd0 --> ' + self.cmd0) def git_pull(self):
"""
更新git内容
:return:
"""
cmd1 = 'git fetch --all;'
cmd2 = 'git reset --hard origin/haoqiao1.3;'
cmd3 = 'git -C {} pull origin haoqiao1.3'.format(self.hotel_fares_dir)
self.exec_cmd(cmd1 + cmd2 + cmd3) def exec_cmd(self, cmd):
"""
执行cmd命令
:param cmd: 需要执行的命令
:return:
:type cmd : str
"""
self.logger.debug('要执行的命令是: ' + cmd)
stdin, stdout, stderr = self.ssh.exec_command(self.cmd0 + cmd)
stdout_str = stdout.read().decode('utf8')
stderr_str = stderr.read().decode('utf8')
if stdout_str != '':
self.logger.info('执行 {} 命令的stdout是 -- > \n{}'.format(cmd, stdout_str))
if stderr_str != '':
self.logger.error('执行 {} 命令的stderr是 -- > \n{}'.format(cmd, stderr_str))
return stdout_str, stderr_str def run_pyhton_file(self, python_file_path):
"""
:param python_file_path:python文件的相对路径
:return:
:type python_file_path:str
"""
python_file_path_short = python_file_path.split('/')[-1] # 防止是直接去进内层文件夹手动启动的短命令
stdout_str, stderr_str = self.exec_cmd("ps aux | grep %s|grep -v grep|awk '{print $2}'" % python_file_path_short)
pid_before = stdout_str
self.logger.info('进程名 [{}] 部署前的进程号是 --> {}'.format(python_file_path_short, pid_before)) self.exec_cmd("ps aux|grep %s|grep -v grep|awk '{print $2}' |xargs kill -9" % python_file_path_short)
self.exec_cmd('nohup python3 %s >/dev/nohup.out 2>&1 &' % python_file_path) # 这里要用长路径,不然找不到文件
# cmd = 'ps aux | grep %s|grep -v grep' % python_file_path
stdout_str, stderr_str = self.exec_cmd("ps aux | grep %s|grep -v grep|awk '{print $2}'" % python_file_path_short)
pid_after = stdout_str
self.logger.info('进程名 [{}] 部署后的进程号是 --> {}'.format(python_file_path_short, pid_after))
if pid_after == '':
self.logger.critical('重新启动 [{}] 失败'.format(python_file_path_short))
elif pid_before != pid_after:
self.logger.warning('重新启动 [{}] 成功'.format(python_file_path_short))
else:
self.logger.critical('启动前后进程号一样'.format(python_file_path_short)) def recover_screen(self):
"""进入screen"""
raise NotImplementedError('还没写,不需要用screen') def run_redis_hotel_id_task_manager(self):
self.run_pyhton_file('app/apis/elongin/elong_proj/redis_hotel_id_task_manager.py') def run_gunicorn(self, application_name, port):
"""
部署gunicorn
:param application_name: 应用名称
:param port: 端口
:return:
"""
application_name_short = application_name.split('.')[-1]
stdout_str, stderr_str = self.exec_cmd("ps aux | grep %s|grep -v grep|awk '{print $2}'" % application_name_short)
pid_before = stdout_str
self.logger.info('进程名 [{}] 部署前的进程号是 --> {}'.format(application_name_short, pid_before)) self.exec_cmd("ps aux|grep %s|grep -v grep|awk '{print $2}' |xargs kill -9" % application_name_short)
self.exec_cmd('nohup gunicorn -w 9 -k gevent --bind 0.0.0.0:%s %s >/dev/webnohup.out 2>&1 &' % (port, application_name)) stdout_str, stderr_str = self.exec_cmd("ps aux | grep %s|grep -v grep|awk '{print $2}'" % application_name_short)
pid_after = stdout_str
self.logger.info('进程名 [{}] 部署后的进程号是 --> {}'.format(application_name_short, pid_after))
if pid_after == '':
self.logger.critical('重新启动 [{}] 失败'.format(application_name_short))
elif pid_before != pid_after:
self.logger.warning('重新启动 [{}] 成功'.format(application_name_short))
else:
self.logger.critical('启动前后进程号一样'.format(application_name_short)) def query_process(self, process_name):
"""查询进程"""
cmd = "ps aux|grep %s|grep -v grep" % process_name
self.exec_cmd(cmd)
python使用paramiko自动化部署linux程序的更多相关文章
- Python学习笔记—自动化部署【新手必学】
前言本文的文字及图片来源于网络,仅供学习.交流使用,不具有任何商业用途,版权归原作者所有,如有问题请及时联系我们以作处理.作者:唯恋殊雨 目录 pexpect fabric pexpect P ...
- 前端自动化部署linux centOs + Jenkins + nignx + 单页面应用
Jenkins是什么? Jenkins 是一款业界流行的开源持续集成工具,广泛用于项目开发,具有自动化构建.测试和部署等功能. 准备工作 Linux centOS系统阿里云服务器一个 码云一个存放vu ...
- pxe+kickstart 自动化部署linux操作系统
kickstart 是什么? 批量部署Linux服务器操作系统 运行模式: C/S client/server 服务器上要部署: DHCP tftp(非交互式文件共享) 安装系统的三个步骤: 1.加载 ...
- Jenkins自动化部署.netcore程序
一.安装jenkins 百度一下 二.构建前的准备 搭建好.net core2.0的环境,下载:https://aka.ms/dotnetcore-2-windowshosting (,net co ...
- Jenkins自动化部署.net程序
一.安装Jenkins 百度上一大堆就不做说明了. 二.构建.net前的准备 1.安装MSBUILD.EXE插件 1.1.进去jenkins->系统管理->插件管理 1.2.配置MSBUI ...
- Jenkins自动化部署入门详细教程
大纲 1.背景 在实际开发中,我们经常要一边开发一边测试,当然这里说的测试并不是程序员对自己代码的单元测试,而是同组程序员将代码提交后,由测试人员测试: 或者前后端分离后,经常会修改接口,然后重新部署 ...
- python+paramiko库+svn写的自动化部署脚本
第一篇博文 直接开门见山的说了. 这是件什么事?:每次部署都是复制本地的文件粘贴到服务器端,因为路径复杂,所以费时且手工容易出漏洞. 一直在想有什么办法可以解决这种,因为以前在微软的一个牛人同事做过一 ...
- Linux 自动化部署
1.pexpect Pexpect 是 Don Libes 的 Expect 语言的一个 Python 实现,是一个用来启动子程序,并使用正则表达式对程序输出做出特定响应,以此实现与其自动交互的 Py ...
- Linux上天之路(十八)之自动化部署
pexpect Pexpect 是 Don Libes 的 Expect 语言的一个 Python 实现,是一个用来启动子程序,并使用正则表达式对程序输出做出特定响应,以此实现与其自动交互的 Pyth ...
随机推荐
- STM32f103的数电采集电路的双ADC的设计与使用
STM32F103C8T6拥有3个ADC,其独立使用已经在本文的3.1.3里面有详细的介绍,这里主要是介绍双ADC的同时使用,即STM32的同步规则模式使用.在此模式在规则通道组上执行时,外部触发来自 ...
- window.print()局部打印三种方式
首先准备要打印的内容,也可以打印时再填充,html中定义如下: <!--startprint--> <div id="printcontent" style=&q ...
- android 中遇到 imageView getWidth 始终为0 时 ,设置 setImageBitmap 的方法
先说说我的遇到的问题: 1. 我在activity里写一个 fragment 2.这个fragment里有个 imageView ,用于显示图片. 我使用 asyncTask获得图片,并准备在这个im ...
- linq操作符:聚合操作符
一.Aggregate操作符 Aggregate操作符对集合值执行自定义聚合运算.来看看Aggregate的定义: public static TSource Aggregate<TSource ...
- 微信小程序 --- e.currentTarget.dataset.id 获取不到值
直接代码 wxml代码片段 <view class='ranksList' wx:for="{{ranksLb}}"> <view class='ranksLis ...
- Java设计模式(6)桥模式(Bridge模式)
Bridge定义:将抽象和行为划分开来,各自独立,但能动态的结合. 为什么使用桥模式 通常,当一个抽象类或接口有多个具体实现(concrete subclass),这些concrete之间关系可能有以 ...
- 从Java开发者的视角解释JavaScript
我们无法在一篇博文里解释JavaScript的所有细节.如果你正或多或少地涉及了web应用程序开发,那么,我们的Java工具和技术范围报告揭示了,大多数(71%)Java开发者被归到了这一类,只是你对 ...
- Android学习之SQLite基础
1.新建MySQLiteHelper类继承自SQLiteOpenHelper public class MySQLiteHelper extends SQLiteOpenHelper { privat ...
- OpenGL 阴影之Shadow Mapping和Shadow Volumes
先说下开发环境.VS2013,C++空项目,引用glut,glew.glut包含基本窗口操作,免去我们自己新建win32窗口一些操作.glew使我们能使用最新opengl的API,因winodw本身只 ...
- logbook日志系统
python中替代logging的日志系统. 不过比之前的logging难理解. 先上打印到屏幕上的代码和存到日志文件中的代码: #!/usr/bin/env python3 # -*- coding ...