关于python:如何判断我的Django应用程序是否在开发服务器上运行?

How can I tell whether my Django application is running on development server or not?

我怎样才能确定我的应用程序是否在开发服务器上运行? 我想我可以检查settings.DEBUG的值并假设DEBUGTrue然后它在开发服务器上运行,但我更愿意知道,而不是依赖于约定。


我在settings.py中添加以下内容以区分标准开发服务器和生产:

1
2
import sys
RUNNING_DEVSERVER = (len(sys.argv) > 1 and sys.argv[1] == 'runserver')

然而,这也依赖于惯例。

(根据Daniel Magnusson的评论修改)


1
2
3
server = request.META.get('wsgi.file_wrapper', None)
if server is not None and server.__module__ == 'django.core.servers.basehttp':
    print('inside dev')

当然,wsgi.file_wrapper可能在META上设置,并且在另一个服务器环境中具有来自名为django.core.servers.basehttp的模块的类极其重合,但我希望这将覆盖您。

顺便说一句,我发现这是通过在开发服务器上运行时创建一个无效的模板,并在TracebackRequest information部分搜索有趣的东西,所以我只是编辑我的答案来证实Nate的想法。


通常我设置一个名为environment的变量并将其设置为"DEVELOPMENT","STAGING"或"PRODUCTION"。在设置文件中,我可以添加基本逻辑,以根据环境更改正在使用的设置。

编辑:此外,您可以简单地使用此逻辑来包含覆盖基本设置的不同settings.py文件。例如:

1
2
if environment =="DEBUG":
    from debugsettings import *

通常这有效:

1
2
3
4
import sys

if 'runserver' in sys.argv:
    # you use runserver

依赖settings.DEBUG是AFAICS最优雅的方式,因为它偶尔也会在Django代码库中使用。

我想你真正想要的是一种自动设置该标志的方法,而无需在每次将项目上传到生产服务器时手动更新它。

为此,我检查settings.py的路径(在settings.py中)以确定项目运行的服务器:

1
2
3
4
5
6
if __file__ =="path to settings.py in my development machine":
    DEBUG = True
elif __file__ in [paths of production servers]:
    DEBUG = False
else:
    raise WhereTheHellIsThisServedException()

请注意,您可能也更喜欢使用@Soviut建议的环境变量进行检查。但是,正如在Windows上开发并在Linux上进行服务检查文件路径的人要比使用环境变量更容易。


如果要根据运行时环境自动切换设置文件
你可以使用环境不同的东西,例如

1
2
3
from os import environ
if environ.get('_', ''):
    print"This is dev - not Apache mod_wsgi"


我刚才遇到了这个问题,最后写了一个类似于Aryeh Leib Taurog的解决方案。我的主要区别在于我想在运行服务器时区分生产环境和开发环境,还要为我的应用程序运行一些一次性脚本(我的运行方式如DJANGO_SETTINGS_MODULE = settings python [脚本])。在这种情况下,只需查看argv [1] == runserver是否足够。所以我想到的是在运行devserver时传递一个额外的命令行参数,当我运行我的脚本时,只需在settings.py中查找该参数。所以代码看起来像这样:

1
2
3
4
5
6
if '--in-development' in sys.argv:
    ## YES! we're in dev
    pass
else:
    ## Nope, this is prod
    pass

然后,运行django服务器成为

python manage.py runserver [你想要的任何选项] - 开发中

并运行我的脚本就像

DJANGO_SETTINGS_MODULE =设置python [myscript] - 开发

只要确保你传递的额外参数不会与任何django冲突(实际上我使用我的应用程序的名称作为参数的一部分)。
我认为这是相当不错的,因为它让我可以准确控制我的服务器和脚本何时表现为prod或dev,并且我不依赖于任何其他人的约定,而不是我自己的约定。

编辑:manage.py抱怨如果你传递无法识别的选项,所以你需要更改settings.py中的代码类似于

1
2
3
if sys.argv[0] == 'manage.py' or '--in-development' in sys.argv:
    # ...
    pass

虽然这有效,但我认识到这不是最优雅的解决方案......


我用:

1
2
3
4
5
DEV_SERVERS = [
    'mymachine.local',
]

DEVELOPMENT = platform.node() in DEV_SERVERS

这需要注意机器上.node()返回的内容。默认情况下,非开发非常重要,这样您就不会意外地暴露敏感的开发信息。

您还可以研究更复杂的方法来唯一识别计算机。


开发和部署环境之间的一个区别将是它正在运行的服务器。具体的不同取决于您的开发和部署环境。

了解您自己的开发和部署环境后,可以使用HTTP请求变量来区分这两者。查看请求变量,如request.META.HTTP_HOSTrequest.META.SERVER_NAMErequest.META.SERVER_PORT,并在两个环境中进行比较。

我打赌你会找到一些非常明显的东西,可以用来检测你的开发环境。在settings.py中进行测试并设置一个可以在别处使用的变量。


settings.DEBUG可以是True并在Apache或其他非开发服务器下运行。它仍然会运行。据我所知,在运行时环境中没有任何内容可以检查pid并与操作系统中的pids进行比较,而这些pid将为您提供此信息。


您可以确定是否在WSGI(mod_wsgi,gunicorn,女服务员等)与manage.py(runserver,test,migrate等)或其他任何内容下运行:

1
2
import sys
WSGI = 'django.core.wsgi' in sys.modules


受Aryeh的回答启发,我为自己设计的技巧就是在sys.argv[0]中查找管理脚本的名称:

1
USING_DEV_SERVER ="pulpdist/manage_site.py" in sys.argv[0]

(我的用例是在运行测试服务器时自动启用Django本机身份验证 - 在Apache下运行时,即使在开发服务器上,我当前项目的所有身份验证都通过Kerberos处理)


你可以检查request.META["SERVER_SOFTWARE"]值:

1
2
3
dev_servers = ["WSGIServer","Werkzeug"]
if any(server in request.META["SERVER_SOFTWARE"] for server in dev_servers):
    print("is local")