10

我有一个问题,我真的不知道从哪里开始解决。也许它会敲响某人的钟声。

TLDR: Django 应用程序崩溃并在重新启动时运行但无法导入某些模块。再次重新启动后,一切都很好。

整个故事:

不同 Python(2.5.x、2.6.x 和 2.6.x)和 Django 版本(分别为 1.1.0、1.2.5 和 1.3.0)上的不同应用程序(我们现在最多三个)偶尔会出现虚假 ImportErrors . 例如,其中一个应用程序通过在内部抛出 ImportError 开始使每个请求失败:

from django.contrib.gis.maps.google import GMarker, GEvent

我们收集了strace输出,相关块如下(为了简洁和保护有罪者,绝对路径替换为 DIR)。

stat64("DIR/django/contrib/gis/maps/google/GMarker", 0xf699ce3c) = -1 ENOENT (No such file or directory)
open("DIR/django/contrib/gis/maps/google/GMarker.so", O_RDONLY|O_LARGEFILE) = -1 ENOENT (No such file or directory)
open("DIR/django/contrib/gis/maps/google/GMarkermodule.so", O_RDONLY|O_LARGEFILE) = -1 ENOENT (No such file or directory)
open("DIR/django/contrib/gis/maps/google/GMarker.py", O_RDONLY|O_LARGEFILE) = -1 ENOENT (No such file or directory)
open("DIR/django/contrib/gis/maps/google/GMarker.pyc", O_RDONLY|O_LARGEFILE) = -1 ENOENT (No such file or directory)

(又是同样的事情 s/GMarker/GEvent/)

重新启动该过程后,一切运行顺利,也在运行:

python -c 'from django.contrib.gis.maps.google import GMarker'

不会产生错误。

GMarker 和 GEvent 类实际上是在 中定义django.contrib.gis.maps.google.overlays和导入的...maps/google/__init__.py

from django.contrib.gis.maps.google.gmap import GoogleMap, GoogleMapSet
from django.contrib.gis.maps.google.overlays import GEvent, GIcon, GMarker, GPolygon, GPolyline
from django.contrib.gis.maps.google.zoom import GoogleZoom

因此完全可以预期加载 GMarker.py 等。将失败。似乎 Python 不知何故忘记了__init__.py它的命名空间。

这些应用程序的流量相对较高,可以想象(尽管不确定)它们可能已经超过了它们的 VM 限制并且几乎可以优雅地恢复。此外,至少在两种情况下,该应用程序存在导致崩溃的早期问题——一种情况下是 SIGSEGV,另一种情况下是错误的……其他情况)。一个应用程序重启导致它抛出 ImportErrors,另一个让它再次运行。损坏的 .py[c]?时间戳是古老的。

所有这些应用程序都在flup wsgi-to-fastcgi 服务器上运行。

到目前为止,这些应用程序中的每一个都失败了一次(在完全不同的模块中,有两种__init__.py“被遗忘”的情况,但我找不到第三个错误 ATM),所以我无法判断这些模块是否有意义。

任何和所有的指针和想法表示赞赏!

4

3 回答 3

1

实际上,您的 strace 线并没有帮助。这些访问不会导致模块被导入。

此类导入错误可能有多种原因:

  1. 有人修改了 sys.path 并且不再在模块路径中
  2. 这些模块中的循环导入,这是由您首先从另一个模块导入触发的!
  3. 你有站点冲突(是的,有那种),从不同的站点包位置加载相同的模块
  4. 您在搜索路径中有一个模块与某处的其他模块或系统模块冲突。

如果您粘贴第二组 strace 行,我们会更接近解决方案吗?

更新

对于 2. 我的意思是,如果您有 2 个具有以下结构的文件

foo.py/初始化.py:

from bar import baz

bar.py/初始化.py:

import foo
def baz():
     pass

snafu.py:

import bar
import foo

好的.py:

import foo
import snafu

运行 python snafu.py 你会得到一个崩溃和类似的 strace 输出,运行 python ok.py 并且一切正常。

于 2012-05-09T18:33:27.363 回答
0

strace 输出对我来说似乎很可疑:

DIR/django/contrib/gis/...

我想知道这DIR部分。您是否有可能在某处输入了 PYTHONPATH 变量DIR而不是$DIR

于 2012-05-09T18:23:40.933 回答
0

我建议您尝试包装导入...除了 ImportError 并添加执行类似操作的代码(任何记录而不是打印的内容都可以)

import sys
print sys.modules["django.contrib.gis.maps.google"]
print dir(sys.modules["django.contrib.gis.maps.google"])

这应该让您了解正在发生的事情。

于 2012-05-23T14:58:31.023 回答