项目场景:

Gunicorn 绿色独角兽’是一个Python WSGI UNIX的HTTP服务器。

在Ubuntu20.04.1上想开机启动python项目,
首先想到的在 /etc/init.d/ 下加了一个custom.sh脚本,然后给执行权限,再加到启动脚本里。

#简单操作命令
cd  /etc/init.d/ 
sudo vim custom.sh
sudo chmod 755 /etc/init.d/custom.sh
sudo update-rc.d custom.sh defaults 90 

#后面失败了就删除了,查了/var/log/syslog没有搜到custom相关信息
sudo update-rc.d -f custom.sh remove

解决方案:

最后想到用crontab定时执行脚本,然后在脚本内简单判断是否启动过,没有则执行启动命令,还别说真好用

crontab设置每5分钟去执行脚本

#crontab内容

*/5 *   * * * vagrant python3 /vagrant/python/cronStartServer.py start
# -*- coding: utf-8 -*-
#python脚本代码

import sys
import os
argv = sys.argv[1]

def start():
    python_pid_list = os.popen("pgrep python").readlines() #pgrep 会把当前脚本进程查出来,pid最大的一个
    print(python_pid_list)
    num = len(python_pid_list)

    if num == 1:
        #开始使用os.popen("xxx").readlines(),运行结果一直卡在这一步,而且没有启动进程
        start_server_log = os.system("nohup python3 /vagrant/python/runApp.py &")
        print(start_server_log)

def stop():
    python_pid_list = os.popen("pgrep python").readlines() 

    print(python_pid_list)
    num = len(python_pid_list)
    if num != 1:
        index = 1
        for i in python_pid_list:
            if index == num:
                break
            #print(type(i))
            os.popen("kill -9 " + i).readlines()
            index += 1


if argv == 'start':
    start()
    print('启动成功')
    print(os.system("sleep 3 && ps aux|grep /vagrant/python/runApp.py |grep -v 'grep'"))
elif  argv == 'stop':
    stop()
    print('停止成功')
elif  argv == 'restart':
    stop()
    start()
    print('重启成功')
    print(os.system("sleep 3 && ps aux|grep /vagrant/python/runApp.py |grep -v 'grep'"))

简单介绍一下python执行系统命令的几种方法
一、os.system方法
os.system(cmd)

在子终端运行系统命令,可以获取命令执行后的返回信息以及执行返回的状态

import os
os.system(‘date’)
2018年 4月 8日 星期日 19时29分13秒 CST
0 #运行状态号,0表示正确
执行后返回两行结果,第一行是结果, 第二行是执行状态信息

二、os.popen方法
os.popen(cmd)

不仅执行命令而且返回执行后的信息对象(常用于需要获取执行命令后的返回信息),是通过一个管道文件将结果返回

import os
nowtime = os.popen(‘date’)
print(nowtime.read())
2018年 4月 8日 星期日 19时30分35秒 CST

三、commands模块
方法 说明

getoutput 获取执行命令后的返回信息

getstatus 获取执行命令的状态值(执行命令成功返回数值0,否则返回非0)

getstatusoutput 获取执行命令的状态值以及返回信息

import commonds
status, output = commands.getstatusoutput(‘date’)
print(status) # 0
print(output) # 2018年 4月 8日 星期日 19时31分45秒 CST
注意1:在类unix的系统下使用此方法返回的返回值(status)与脚本或命令执行之后的返回值不等,这是因为调用了os.wait()的缘故,具体原因就得去了解下系统wait()的实现了。需要正确的返回值(status),只需要对返回值进行右移8位操作就可以了。

注意2:当执行命令的参数或者返回中包含了中文文字,那么建议使用subprocess。

四、subprocess模块
运用对线程的控制和监控,将返回的结果赋于一变量,便于程序的处理。有丰富的参数可以进行配置,可供我们自定义的选项多,灵活性高。之前我使用os.system的时候遇到文件描述符被子进程继承的问题,后来通过close_fds = False 这个参数来解决的。官方文档:http://python.usyiyi.cn/python_278/library/subprocess.html

import subprocess
nowtime = subprocess.Popen(‘date’, shell=True, stdout=subprocess.PIPE, stderr=subprocess.STDOUT)
print(nowtime.stdout.read())
2018年 4月 8日 星期日 19时32分41秒 CST