1. Day 13 - 提升开发效率
    2. 阅读: 12834
    3. 现在,我们已经把一个Web App的框架完全搭建好了,从后端的API到前端的MVVM,流程已经跑通了。
    4. 在继续工作前,注意到每次修改Python代码,都必须在命令行先Ctrl-C停止服务器,再重启,改动才能生效。
    5. 在开发阶段,每天都要修改、保存几十次代码,每次保存都手动来这么一下非常麻烦,严重地降低了我们的开发效率。有没有办法让服务器检测到代码修改后自动重新加载呢?
    6. Django的开发环境在Debug模式下就可以做到自动重新加载,如果我们编写的服务器也能实现这个功能,就能大大提升开发效率。
    7. 可惜的是,Django没把这个功能独立出来,不用Django就享受不到,怎么办?
    8. 其实Python本身提供了重新载入模块的功能,但不是所有模块都能被重新载入。另一种思路是检测www目录下的代码改动,一旦有改动,就自动重启服务器。
    9. 按照这个思路,我们可以编写一个辅助程序pymonitor.py,让它启动wsgiapp.py,并时刻监控www目录下的代码改动,有改动时,先把当前wsgiapp.py进程杀掉,再重启,就完成了服务器进程的自动重启。
    10. 要监控目录文件的变化,我们也无需自己手动定时扫描,Python的第三方库watchdog可以利用操作系统的API来监控目录文件的变化,并发送通知。我们先用pip安装:
    11. $ pip3 install watchdog
    12. 利用watchdog接收文件变化的通知,如果是.py文件,就自动重启wsgiapp.py进程。
    13. 利用Python自带的subprocess实现进程的启动和终止,并把输入输出重定向到当前进程的输入输出中:
    14. #!/usr/bin/env python3
    15. # -*- coding: utf-8 -*-
    16. __author__ = 'Michael Liao'
    17. import os, sys, time, subprocess
    18. from watchdog.observers import Observer
    19. from watchdog.events import FileSystemEventHandler
    20. def log(s):
    21. print('[Monitor] %s' % s)
    22. class MyFileSystemEventHander(FileSystemEventHandler):
    23. def __init__(self, fn):
    24. super(MyFileSystemEventHander, self).__init__()
    25. self.restart = fn
    26. def on_any_event(self, event):
    27. if event.src_path.endswith('.py'):
    28. log('Python source file changed: %s' % event.src_path)
    29. self.restart()
    30. command = ['echo', 'ok']
    31. process = None
    32. def kill_process():
    33. global process
    34. if process:
    35. log('Kill process [%s]...' % process.pid)
    36. process.kill()
    37. process.wait()
    38. log('Process ended with code %s.' % process.returncode)
    39. process = None
    40. def start_process():
    41. global process, command
    42. log('Start process %s...' % ' '.join(command))
    43. process = subprocess.Popen(command, stdin=sys.stdin, stdout=sys.stdout, stderr=sys.stderr)
    44. def restart_process():
    45. kill_process()
    46. start_process()
    47. def start_watch(path, callback):
    48. observer = Observer()
    49. observer.schedule(MyFileSystemEventHander(restart_process), path, recursive=True)
    50. observer.start()
    51. log('Watching directory %s...' % path)
    52. start_process()
    53. try:
    54. while True:
    55. time.sleep(0.5)
    56. except KeyboardInterrupt:
    57. observer.stop()
    58. observer.join()
    59. if __name__ == '__main__':
    60. argv = sys.argv[1:]
    61. if not argv:
    62. print('Usage: ./pymonitor your-script.py')
    63. exit(0)
    64. if argv[0] != 'python3':
    65. argv.insert(0, 'python3')
    66. command = argv
    67. path = os.path.abspath('.')
    68. start_watch(path, None)
    69. 一共70行左右的代码,就实现了Debug模式的自动重新加载。用下面的命令启动服务器:
    70. $ python3 pymonitor.py wsgiapp.py
    71. 或者给pymonitor.py加上可执行权限,启动服务器:
    72. $ ./pymonitor.py app.py
    73. 在编辑器中打开一个.py文件,修改后保存,看看命令行输出,是不是自动重启了服务器:
    74. $ ./pymonitor.py app.py
    75. [Monitor] Watching directory /Users/michael/Github/awesome-python3-webapp/www...
    76. [Monitor] Start process python app.py...
    77. ...
    78. INFO:root:application (/Users/michael/Github/awesome-python3-webapp/www) will start at 0.0.0.0:9000...
    79. [Monitor] Python source file changed: /Users/michael/Github/awesome-python-webapp/www/handlers.py
    80. [Monitor] Kill process [2747]...
    81. [Monitor] Process ended with code -9.
    82. [Monitor] Start process python app.py...
    83. ...
    84. INFO:root:application (/Users/michael/Github/awesome-python3-webapp/www) will start at 0.0.0.0:9000...
    85. 现在,只要一保存代码,就可以刷新浏览器看到效果,大大提升了开发效率。