python启动脚本--daemon形式

这篇博客介绍如何用Python编写一个具备启动、停止、重启、状态查询和帮助功能的守护进程脚本,并提供了具体的代码示例。通过这个脚本,可以将程序以daemon形式运行。

和上一篇实现同样的start,stop,restart,status,help等功能,不同的是以daemon形式,直接贴代码:

[root@js python]# cat rc.memcached1.py
#!/usr/bin/env python
import sys
from subprocess import Popen,PIPE
import os

class Process(object):
    '''memcached rc scripts'''
    args = {'USER':'memcached',
            'PORT':11211,
            'MAXCONN':1024,
            'CACHESIZE':64,
            'OPTIONS':''}
    def __init__(self,name,program,workdir):
        self.name = name
        self.program = program
        self.workdir = workdir

    def _init(self):
        '''/var/tmp/memcached'''
        if not os.path.exists(self.workdir):
            os.mkdir(self.workdir)
            os.chdir(self.workdir)

    def _pidFile(self):
        '''/var/tmp/memcached/memcached.pid '''
        return os.path.join(self.workdir,"%s.pid" %self.name)

    def _writePid(self):
        if self.pid:
            with open(self._pidFile(),'w') as fd:
                fd.write(str(self.pid))
    def _readConf(self,f):
        with open(f) as fd:
            lines = fd.readlines()
            return dict([i.strip().replace('"','').split('=') for i in lines])

    def _parseArgs(self):
        conf = self._readConf('/etc/sysconfig/memcached')
        if 'USER' in conf:
            self.args['USER'] = conf['USER']
        if 'PORT' in conf:
            self.args['PORT'] = conf['PORT']
        if 'MAXCONN' in conf:
            self.args['MAXCONN'] = conf['MAXCONN']
        if 'CACHESIZE' in conf:
            self.args['CACHESIZE'] = conf['CACHESIZE']
        op = ['-u',self.args['USER'],'-p',self.args['PORT'],'-m',self.args['CACHESIZE'],'-c',self.args['MAXCONN']]
        return op


    def start(self):
        pid = self._getPid()
        if pid:
            print "%s is already running" %self.name
            sys.exit()
        self._init()
        os.system("chown %s %s" %(self.args['USER'],self.workdir))

        cmd = [self.program] + self._parseArgs() + ['-d','-P',self._pidFile()]
        print cmd
        p = Popen(cmd,stdout=PIPE)
        #self.pid = p.pid
        #self._writePid()
        print "%s is success" %self.name
    def _getPid(self):
        p = Popen(['pidof',self.name],stdout=PIPE)
        pid = p.stdout.read().strip()
        return pid
    def stop(self):
        pid = self._getPid()
        if pid:
            os.kill(int(pid),15)
            if os.path.exists(self._pidFile()):
                os.remove(self._pidFile())
            print "%s is stopped" %self.name
    def restart(self):
        self.stop()
        self.start()

    def status(self):
        pid = self._getPid()
        if pid:
            print "%s is running" %self.name
        else:
            print "%s is stopped" %self.name
    def help(self):
        print "Usage:%s {start|stop|restart|status}" %__file__
def main():
    name = 'memcached'
    program = '/usr/bin/memcached'
    workdir = '/var/tmp/memcached'
    pm =Process(name=name,
                program=program,
                workdir=workdir)
    try:
        cmd = sys.argv[1]
    except IndexError,e:
        print "option error"
        sys.exit()
    if cmd == 'start':
        pm.start()
    elif cmd == 'stop':
        pm.stop()
    elif cmd == 'restart':
        pm.restart()
    elif cmd == 'status':
        pm.status()
    else:
        pm.help()
if __name__ == '__main__':
    main()

测试一下:


[root@js python]# python rc.memcached1.py  start
['/usr/bin/memcached', '-u', 'memcached', '-p', '11211', '-m', '64', '-c', '1024', '-d', '-P', '/var/tmp/memcached/memcached.pid']
memcached is success
[root@js python]# ps -ef|grep memcached
496        1828      1  0 07:47 ?        00:00:00 /usr/bin/memcached -u memcached -p 11211 -m 64 -c 1024 -d -P /var/tmp/memcached/memcached.pid
root       1835   1497  0 07:47 pts/0    00:00:00 grep memcached
[root@js python]# cat /var/tmp/memcached/memcached.pid
1828
[root@js python]# ls -ld !$
ls -ld /var/tmp/memcached/memcached.pid
-rw-r--r-- 1 memcached memcached 5 716 07:47 /var/tmp/memcached/memcached.pid
[root@js python]# python rc.memcached1.py  stop
memcached is stopped
[root@js python]# cat /var/tmp/memcached/memcached.pid
cat: /var/tmp/memcached/memcached.pid: 没有那个文件或目录
[root@js python]# python rc.memcached1.py  status
memcached is stopped

结束。

### Python 守护进程库的使用方法 守护进程是一种在后台运行的服务程序,在 Unix 和类 Unix 系统上非常常见。为了创建守护进程,可以利用 `python-daemon` 库来简化这一过程。 #### 创建简单的守护进程实例 安装 `python-daemon` 可通过 pip 实现: ```bash pip install python-daemon ``` 下面是一个简单例子展示如何编写一个守护进程[^1]: ```python import daemon import time from datetime import datetime def run(): while True: with open("/tmp/daemon.log", "a") as f: now = datetime.now().strftime('%Y-%m-%d %H:%M:%S') f.write(f"{now}\n") time.sleep(5) with daemon.DaemonContext(): run() ``` 这段代码会每五秒向 `/tmp/daemon.log` 文件追加当前时间戳记录。当此脚本被执行时,它会在后台持续运行直到被手动终止。 对于更复杂的场景,可能还需要配置日志文件位置、工作目录以及用户权限等参数。这些都可以通过传递给 `DaemonContext()` 的关键字参数完成设置。 #### 配置高级选项 如果希望进一步定制化守护进程的行为,则可以在初始化 `DaemonContext` 对象时指定更多属性。例如: ```python context = daemon.DaemonContext( working_directory='/var/lib/myapp', umask=0o002, pidfile=pidlockfile.TimeoutPIDLockFile('/var/run/mydaemon.pid'), ) with context: main_loop_function() ``` 上述片段展示了怎样设定工作路径 (`working_directory`) ,改变新进程中文件模式位掩码(`umask`) 并且管理 PID 锁定文件 (`pidfile`). 这样做有助于确保守护进程按照预期方式启动并保持稳定状态。 #### 处理信号量 有时也需要让守护进程响应特定操作系统级别的事件或命令,比如优雅停机。这可以通过捕获相应信号实现: ```python import signal def terminate_handler(signum, frame): sys.exit() signal.signal(signal.SIGTERM, terminate_handler) ``` 这里定义了一个处理函数用于接收到 SIGTERM 信号后安全退出应用程序。 #### 注意事项 值得注意的是,尽管以上介绍的方法适用于大多数情况下的守护进程开发需求;但在某些特殊环境中(如容器内),由于缺乏完整的 Unix 功能支持,可能会遇到一些挑战。因此建议开发者根据实际应用场景调整方案设计。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值