Django框架 - 是否有可以订阅的关闭事件?

时间:2013-03-18 08:00:27

标签: python django web

我有一个问题,我希望可以通过Django中的某种形式的关闭钩子来解决。

我仍然是Python / Django开发的初学者,为了帮助我学习,我已经为自己设定了一个开发在浏览器中运行的COMET / Reverse Ajax风格聊天网站的项目。浏览器不断轮询服务器以查找任何消息。当服务器收到轮询请求时,它会检查是否有任何消息在等待,如果没有,它将尝试获取已经获取的 threading.Lock 对象的锁定。这会导致处理请求的线程阻塞,直到收到消息并释放前面提到的锁定。

当我关闭服务器时出现问题。此时我正在使用开发服务器(python manage.py runserver)。如果上述过程中有一个线程被阻止,则服务器不会关闭。

当服务器尝试关闭以便我可以释放任何等待的线程时,有没有办法可以执行代码?

我查看了以下similar question,但它没有帮助我。有人猜测这种事件挂钩不存在,但没有明确的答案。我已经googled高低,但没有运气。

我正在使用Python 2.7和Django 1.5。

感谢您提供的任何帮助

2 个答案:

答案 0 :(得分:7)

我不知道关闭事件,但您可以创建一个。

当用户按下Ctrl-C时,会向Python进程发送SIGINT信号。从wsgi.py文件设置信号处理程序以捕获SIGINT并将其转发到您的程序。

wsgi.py:

...

import signal
signal.signal(signal.SIGINT, my_signal_handler)

你甚至可以将信号转发到django信号:

my_django_shutdown_signal = django.dispatch.Signal()

def _forward_to_django_shutdown_signal(signal, frame):
    my_django_shutdown_signal.send('system')
    sys.exit(0)   # So runserver does try to exit
signal.signal(signal.SIGINT, _forward_to_django_shutdown_signal)

由于只能从主线程调用signal,因此这不适用于runserver。使用--noreload选项调用runserver,或切换到gunicorn。

答案 1 :(得分:2)

除了Stu Gla的回答:
要在django的主线程中设置信号事件的处理程序,你可以把处理初始化放到manage.py或你的app的__init__.py中,如下所示:

#  {project_root}/{your_app}/__init__.py
import os  
import signal  
import sys  
def my_signal_handler(*args):  
    if os.environ.get('RUN_MAIN') == 'true':  
        print('stopped'.upper())  
    sys.exit(0) 

signal.signal(signal.SIGINT, my_signal_handler)  

它可以让你在没有 - noreload 的情况下运行服务器 检查os.environ.get('RUN_MAIN') pvevent呼叫功能2次。