我看过很多关于 ValueError: relative import in non-packge
的讨论,但是关于调用脚本的讨论似乎与我当前遇到的问题无关(或者如果有的话,我不太明白如何 他们确实...)。
昨天,我设置了一个运行 Apache 的新 Ubuntu Server VM。我有一个 Python3.4 virtualenv,Apache 在运行我的 Django 应用程序时将其用作其解释器和包目录。
我的目录结构如下所示:
目录结构
Django_proj/
manage.py
...
Django_proj/
__init__.py
wsgi.py
urls.py
settings/
__init__.py
core_settings.py
prod_settings.py
local_settings.py
logging_settings.py
<小时/>
目前,我正在加载 local_settings.py
,其中我有以下相对导入:
from .logging_settings import LOGGING
但我在 Apache 日志中得到如下所示的回溯:
File "/PATH/TO/VIRTUALENV/lib/python3.4/site-packages/django/utils/importlib.py", line 40, in import_module
__import__(name)
File "/PATH/TO/Django_proj/Django_proj/local_settings.py", line 12, in <module>
from .logging_settings import LOGGING
ValueError: Attempted relative import in non-package
<小时/>
外壳工程
如果我转到顶级目录并运行(激活 virtualenv 后)python manage.py shell
,则不会出现此错误:它能够成功导入 local_settings.py
和我的任何 django 应用程序,并且我可以访问变量(例如 LOGGING
中的 django.conf.settings
) .
wsgi.py
这是 wsgi.py 文件,我在其中将列出的所有目录添加到路径中:
import os
import sys
sys.path.append('PATH/TO/Django_proj')
sys.path.append('PATH/TO/Django_proj/Django_proj')
sys.path.append('PATH/TO/Django_proj/Django_proj/settings')
os.environ.setdefault("DJANGO_SETTINGS_MODULE",
"settings.local_settings")
from django.core.wsgi import get_wsgi_application
application = get_wsgi_application()
<小时/>
Apache 配置
Apache 成功导入了 Django,并且使用了正确的 Python 解释器(可以在回溯中看到),但我认为 Apache 运行应用程序的方式可能有些奇怪,所以除了 wsgi.py
中的路径之外,我将我能想到的所有内容添加到 WSGIDaemonProcess path
中。这是我的 httpd.conf:
LoadModule wsgi_module modules/mod_wsgi.so
ServerName localhost
WSGIRestrictEmbedded On
WSGILazyInitialization On
WSGIDaemonProcess django-proj.org processes=2 threads=12 python-path=/virtualenv/PATH/lib/python3.4/site-packages:/virtualenv/PATH/bin:/PATH/TO/Django_proj:/PATH/TO/Django_proj/Django_proj:/PATH/TO/Django_proj/Django_proj/settings
WSGIProcessGroup django-proj.org
但是我不太了解 Apache 启动 Python 以运行 wsgi.py
时发生的情况。我想如果我只是在各处添加了所有相关路径,那么问题就会消失,Python 应该能够从 logging_settings.py
本地找到并导入 local_settings.py
。然而,local_settings.py
似乎是一个非包。
无论如何,我很困惑为什么会收到此错误,并且非常感谢知情的 Stack Overflowers 的意见。
编辑:我应该提一下,我也多次阅读了以下文档,它很有用:https://code.google.com/p/modwsgi/wiki/IntegrationWithDjango
最佳答案
嗯,这很尴尬,但我真的不知道我做了什么来解决这个问题。然而,它是固定的。这就是我所做的。
我的从源代码安装的 Python shell 遇到了一个不相关的问题,没有使用 readline(没有历史记录),我按照此处所述修复了该问题:https://stackoverflow.com/a/25942732/1748754
我需要在构建 readline 后重新编译 Python,因为这一切都在 VM 上(使用 Vagrant 和 Ansible 设置),所以我只是销毁了它并重新构建了它。
之后,我注意到在我的 wsgi.py
中,我以一种奇怪的方式导入设置,因此我只是按以下方式附加了主项目路径和导入的设置:
wsgi.py
import os
import sys
sys.path.append('/PATH/TO/Django_proj/Django_proj/')
os.environ.setdefault("DJANGO_SETTINGS_MODULE",
"Django_proj.settings.local_settings")
from django.core.wsgi import get_wsgi_application
application = get_wsgi_application()
我进行此更改是因为我意识到 wgsi.py
和 manage.py
正在以不同的方式导入 settings.py
,所以我将 wsgi.py
更改为 manage.py
。
很难相信 readline 缺失导致了我其他错误,如果是这样,这可能是非常具体的,对其他人没有用。
问题在于,我只是在完成所有其他工作后才想到检查它是否有效。
不管怎样,不知道是什么解决了这个问题,这有点尴尬,它涉及到一个完整的操作系统拆卸和重建(这对于 Vagrant/Ansible 来说非常容易),但事实就是如此。关于python - Django Apache 值错误 : Attempted Relative Import in non-package,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25940464/