在 django 内部和外部运行的包。这是一个好的设计吗?

Pab*_*blo 6 python django

我对 django 以及 python 世界都比较陌生。但我有使用 ruby​​ 的经验(使用 Rails 已经有 2 年了),所以 python/django 的许多概念对我来说并不新鲜。

无论如何,我正在用 python 编写一个小包,它将有一个数据库连接,我想在 django 内部使用这个包,但也在 django 外部的功能中使用。所以我决定利用django.db,而不用担心编写任何数据库连接和管理内容。

所以我开始编写我的第一个模型,并想在 django 环境之外进行第一次测试,但我发现自己遇到了一些困难。

我考虑过为我的包使用与任何 django 应用程序相同的配置机制(我的意思是 settings.py 文件)。我编写了一个仅包含 DATABASES 字典的文件(称为 nodjango_settings.py 作为模板),并向其中添加了两个自定义变量:

MYAPP_DB_ID   = "myappdb"
MYAPP_DB_PREFIX = "myapp_"
DATABASES = {
  MYAPP_DB_ID: {
    'ENGINE': 'django.db.backends.postgresql_psycopg2',
    'NAME':   'myappdb',
  }
}
Run Code Online (Sandbox Code Playgroud)

我的包的目录结构是:

.
|-- README.txt
|-- doc
|   `-- db.txt
|-- setup.py
|-- src
|   |-- __init__.py
|   `-- myapp
|       |-- __init__.py
|       |-- exceptions.py
|       |-- models.py
|       |-- nodjango_settings.py
|       `-- nodjango_settings.pyc
`-- test
    `-- test.py
Run Code Online (Sandbox Code Playgroud)

我正在阅读一些 djangos 自己的代码,以了解如何处理 DJANGO_SETTINGS_MODULE 环境变量并读取配置文件,因此将以下代码添加到myapp/__init__py.

import os

# try to look after the DJANGO_SETTINGS_MODULE environment variable
# if not present raise an import error

# code abstract from python2.6/site-packages/django/conf/__init__.py
try:
    settings_module = os.environ["DJANGO_SETTINGS_MODULE"]
    if not settings_module: # If it's set but is an empty string.
        raise KeyError
except KeyError:
    raise ImportError("The DJANGO_SETTINGS_MODULE environment variable is not present.")


# TODO: look at python2.6/site-packages/django/conf/__init__.py +93
# if you print sys.path then the project directory gets added
# in django/core/management/__init__.py with sys.path.append
from django.utils import importlib

p = importlib.import_module(settings_module)

print p.MYAPP_DB_ID
Run Code Online (Sandbox Code Playgroud)

我想测试它__init__.py是否按预期工作,因此在我运行的包的根目录上:

$ DJANGO_SETTINGS_MODULE="src.myapp.nodjango_settings" python src/myapp/__init__.py


Traceback (most recent call last):
  File "src/myapp/__init__.py", line 22, in <module>
    p = importlib.import_module(settings_module)
  File "/home/yanez/devpython/lib/python2.6/site-packages/django/utils/importlib.py",   line 35, in import_module
    __import__(name)
ImportError: No module named src.myapp.nodjango_settings
Run Code Online (Sandbox Code Playgroud)

我没想到会这样。当您创建新的 Django 应用程序时,manage.py会创建一个文件。并且该文件DJANGO_SETTINGS_MODULE从应用程序根目录的角度将变量设置到settings.py文件中。

我阅读了更多 django 代码并意识到在django/core/management/__init__.py应用程序的根目录中添加了sys.path. 这也可能是我的问题的解决方案,但我不太确定这是否是一个好主意sys.path

我不太确定我的想法是好还是坏。我想知道您对此有何看法以及我可以在哪里/如何改进它。此外,如果我坚持我的想法,如何读取自定义变量而settings.py不必一遍又一遍地重新导入设置模块?

bru*_*ers 3

当使用 mod_wsgi 部署 django 项目时,必须编写一个 wsgi (python) 脚本,该脚本执行“sys.path”舞蹈并设置 DJANGO_SETTINGS_MODULE 环境变量,然后创建 wsgi 应用程序对象等。

我为什么要提到这一点?因为,恕我直言,您不应该尝试在“myapp”中处理这部分问题,而应该从一个不同的 python 脚本来处理,该脚本将是在 django 项目上下文之外使用您的应用程序时的应用程序入口点,并将“myapp”保留为纯库包。然后,该启动器脚本将负责设置正确的环境(sys.path、设置等)。根据记录,设置模块只是一个Python模块,在运行时它是一个普通的Python对象(“模块”类的实例),并且有很多方法(除了默认的导入机制)来创建模块实例并将其添加到 sys.modules (这是这里的重点)。

附带说明一下,恕我直言,在包中进行设置没有意义,它是一个配置文件。

编辑:嗯,我知道有一些关于使用 django 独立部分的内容,这里是:https ://docs.djangoproject.com/en/dev/topics/settings/#using-settings-without-setting-django-设置模块