1

我有一个系统,我需要在请求之前和之后对 URL 进行完全动态控制。

我正在为此使用信号,并且对于预请求信号(我遇到问题的那个,我有一个像这样的中间件,它连接到信号,允许它检查当前 request.path 是否适用到它,然后与它得到的第一个一起去。这通常工作得很好,而且相当优雅):

class PreRouteMiddleWare(object):
    def process_request(self, request):
        url = request.path.strip('/')
        if url == '':
            url = '/'
        pre_routes = pre_route.send(sender=request, url=url)
        for reciever, response in pre_routes:
            if response:
                return response
        return None

现在,要注册在 Django 路由堆栈“之前”发生的事情,我在应用程序中执行以下操作models.py

@receiver(pre_route)
def try_things(sender, url, **kwargs):
    try:
        thing= Thing.objects.get(url=url)
        from myapp.views import myview
        return myview(sender, some_args)
    except Thing.DoesNotExist:
        return False

这在我的开发服务器上也很有效。

但是,问题出现在我使用 uWSGI 的生产环境中。我像这样启动uWSGI(从新贵):

sudo /usr/local/bin/uwsgi --emperor '/srv/*/uwsgi.ini' --enable-threads --single-interpreter

我的 uwsgi.ini 看起来像这样:

[uwsgi]
socket = /srv/new/uwsgi.sock
module = wsgi:app
chdir = /srv/new/myapp
virtualenv = /srv/new
env = DJANGO_SETTINGS_MODULE=myapp.settings
uid = wsgi_new
gid = www-data
chmod = 770
processes = 2

似乎正在发生的是对于每个 uWSGI 进程/线程,它们似乎只在第一个请求时加载 models.py,这意味着每个进程的第一个请求将无法连接信号。这意味着我有 n 个(其中 n 是进程数)请求完全失败,因为 models.py 在启动时没有加载(因为它正在开发中)。

我配置 uWSGI 错了吗?有没有更好的方法来强制在启动时连接信号?

4

2 回答 2

3

Django 实际上是懒惰地加载东西。使用开发服务器会让人产生一种错误的安全感,即事情将如何在真正的 WSGI 服务器中工作,因为开发服务器加载管理命令会强制进行许多生产服务器不会发生的早期初始化。

你可能会读到:

http://blog.dscpl.com.au/2010/03/improved-wsgi-script-for-use-with.html

这解释了在 mod_wsgi 中发生的问题。uWSGI 也会发生同样的事情。

于 2012-06-19T22:24:57.317 回答
0

好的,事实证明我需要让我的中间件挂钩 process_view 而不是 process_request:

class PreRouteMiddleWare(object):
    def process_view(self, request, *args, **kwargs):
        url = request.path.strip('/')
        if url == '':
            url = '/'
        pre_routes = pre_route.send(sender=request, url=url)
        for reciever, response in pre_routes:
            if response:
                return response
        return None

现在效果很好!

于 2012-06-21T12:37:35.740 回答