用Python实现服务器中只重载被修改的进程的方法

作者:廖雪峰 时间:2022-06-21 05:11:38 

现在,我们已经把一个Web App的框架完全搭建好了,从后端的API到前端的MVVM,流程已经跑通了。

在继续工作前,注意到每次修改Python代码,都必须在命令行先Ctrl-C停止服务器,再重启,改动才能生效。

在开发阶段,每天都要修改、保存几十次代码,每次保存都手动来这么一下非常麻烦,严重地降低了我们的开发效率。有没有办法让服务器检测到代码修改后自动重新加载呢?

Django的开发环境在Debug模式下就可以做到自动重新加载,如果我们编写的服务器也能实现这个功能,就能大大提升开发效率。

可惜的是,Django没把这个功能独立出来,不用Django就享受不到,怎么办?

其实Python本身提供了重新载入模块的功能,但不是所有模块都能被重新载入。另一种思路是检测www目录下的代码改动,一旦有改动,就自动重启服务器。

按照这个思路,我们可以编写一个辅助程序pymonitor.py,让它启动wsgiapp.py,并时刻监控www目录下的代码改动,有改动时,先把当前wsgiapp.py进程杀掉,再重启,就完成了服务器进程的自动重启。

要监控目录文件的变化,我们也无需自己手动定时扫描,Python的第三方库watchdog可以利用操作系统的API来监控目录文件的变化,并发送通知。我们先用easy_install安装:


$ easy_install watchdog

利用watchdog接收文件变化的通知,如果是.py文件,就自动重启wsgiapp.py进程。

利用Python自带的subprocess实现进程的启动和终止,并把输入输出重定向到当前进程的输入输出中:


#!/usr/bin/env python
import os, sys, time, subprocess

from watchdog.observers import Observer
from watchdog.events import FileSystemEventHandler

def log(s):
 print '[Monitor] %s' % s

class MyFileSystemEventHander(FileSystemEventHandler):
 def __init__(self, fn):
   super(MyFileSystemEventHander, self).__init__()
   self.restart = fn

def on_any_event(self, event):
   if event.src_path.endswith('.py'):
     log('Python source file changed: %s' % event.src_path)
     self.restart()

command = ['echo', 'ok']
process = None

def kill_process():
 global process
 if process:
   log('Kill process [%s]...' % process.pid)
   process.kill()
   process.wait()
   log('Process ended with code %s.' % process.returncode)
   process = None

def start_process():
 global process, command
 log('Start process %s...' % ' '.join(command))
 process = subprocess.Popen(command, stdin=sys.stdin, stdout=sys.stdout, stderr=sys.stderr)

def restart_process():
 kill_process()
 start_process()

def start_watch(path, callback):
 observer = Observer()
 observer.schedule(MyFileSystemEventHander(restart_process), path, recursive=True)
 observer.start()
 log('Watching directory %s...' % path)
 start_process()
 try:
   while True:
     time.sleep(0.5)
 except KeyboardInterrupt:
   observer.stop()
 observer.join()

if __name__ == '__main__':
 argv = sys.argv[1:]
 if not argv:
   print('Usage: ./pymonitor your-script.py')
   exit(0)
 if argv[0]!='python':
   argv.insert(0, 'python')
 command = argv
 path = os.path.abspath('.')
 start_watch(path, None)

一共50行左右的代码,就实现了Debug模式的自动重新加载。用下面的命令启动服务器:


$ python pymonitor.py wsgiapp.py

或者给pymonitor.py加上可执行权限,启动服务器:


$ ./pymonitor.py wsgiapp.py

在编辑器中打开一个py文件,修改后保存,看看命令行输出,是不是自动重启了服务器:


$ ./pymonitor.py wsgiapp.py
[Monitor] Watching directory /Users/michael/Github/awesome-python-webapp/www...
[Monitor] Start process python wsgiapp.py...
...
INFO:root:application (/Users/michael/Github/awesome-python-webapp/www) will start at 0.0.0.0:9000...
[Monitor] Python source file changed: /Users/michael/Github/awesome-python-webapp/www/apis.py
[Monitor] Kill process [2747]...
[Monitor] Process ended with code -9.
[Monitor] Start process python wsgiapp.py...
...
INFO:root:application (/Users/michael/Github/awesome-python-webapp/www) will start at 0.0.0.0:9000...
Try

标签:Python
0
投稿

猜你喜欢

  • 举例讲解Python中的list列表数据结构用法

    2021-05-07 08:02:53
  • 网站设计应当让用户选,别让用户想

    2008-03-19 12:01:00
  • 非常全面的Python常见基础面试题及答案

    2022-11-14 17:03:33
  • display:inline问题小结

    2008-05-01 13:08:00
  • python自动化测试之从命令行运行测试用例with verbosity

    2021-09-20 07:32:16
  • django自带的server 让外网主机访问方法

    2023-06-03 22:59:50
  • 如何在页面错误时向数据库中添加记录?

    2010-06-26 12:26:00
  • Oracle 11g安装错误提示未找到wfmlrsvcapp.ear的解决方法

    2023-07-14 14:31:52
  • 一个完美网站的101项指标.第七部分.W3C标准

    2008-03-01 09:31:00
  • python的ping网络状态监测的实现(含多IP)

    2022-03-16 07:21:29
  • python人工智能tensorflow常用激活函数Activation Functions

    2023-08-01 12:45:02
  • php中关于普通表单多文件上传的处理方法

    2023-11-14 20:21:21
  • python实现连续变量最优分箱详解--CART算法

    2023-01-15 16:33:05
  • python内存管理机制原理详解

    2021-05-24 16:19:48
  • 一段ASP单页显示文件夹下所有图片的代码

    2011-02-28 11:23:00
  • Python使用Chrome插件实现爬虫过程图解

    2023-08-01 02:02:29
  • Python爬虫自动化爬取b站实时弹幕实例方法

    2023-03-21 04:45:42
  • sp_executesql 使用复杂的Unicode 表达式错误的解决方法

    2012-01-29 17:58:52
  • 浅析DW4中的站点管理

    2007-02-03 11:40:00
  • 一些需要禁用的PHP危险函数(disable_functions)

    2023-11-23 15:29:25
  • asp之家 网络编程 m.aspxhome.com