我有一个问题,我希望可以通过Django中的某种形式的关闭钩子来解决。
我仍然是Python / Django开发的初学者,为了帮助我学习,我已经为自己设定了一个开发在浏览器中运行的COMET / Reverse Ajax风格聊天网站的项目。浏览器不断轮询服务器以查找任何消息。当服务器收到轮询请求时,它会检查是否有任何消息在等待,如果没有,它将尝试获取已经获取的 threading.Lock 对象的锁定。这会导致处理请求的线程阻塞,直到收到消息并释放前面提到的锁定。
当我关闭服务器时出现问题。此时我正在使用开发服务器(python manage.py runserver)。如果上述过程中有一个线程被阻止,则服务器不会关闭。
当服务器尝试关闭以便我可以释放任何等待的线程时,有没有办法可以执行代码?
我查看了以下similar question,但它没有帮助我。有人猜测这种事件挂钩不存在,但没有明确的答案。我已经googled高低,但没有运气。
我正在使用Python 2.7和Django 1.5。
感谢您提供的任何帮助
答案 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次。