为什么 twistd 不能从模块的当前工作目录中导入模块?



请考虑以下测试用例。

项目目录的结构如下:

foo
├── foo
│   ├── __init__.py
│   └── bar.py
└── test.tac

bar.py包含一个简单的类定义:

# bar.py
class Bar:
pass

test.tac,扭曲的应用程序配置文件,包含一个导入语句:

#test.tac
from foo.bar import Bar

运行twistd -ny test.tac时,出现以下错误:

$ twistd -ny test.tac
Unhandled Error
Traceback (most recent call last):
File "/Users/lthibault/.pyenv/versions/3.6.2/lib/python3.6/site-packages/twisted/application/app.py", line 674, in run
runApp(config)
File "/Users/lthibault/.pyenv/versions/3.6.2/lib/python3.6/site-packages/twisted/scripts/twistd.py", line 25, in runApp
runner.run()
File "/Users/lthibault/.pyenv/versions/3.6.2/lib/python3.6/site-packages/twisted/application/app.py", line 381, in run
self.application = self.createOrGetApplication()
File "/Users/lthibault/.pyenv/versions/3.6.2/lib/python3.6/site-packages/twisted/application/app.py", line 453, in createOrGetApplication
application = getApplication(self.config, passphrase)
--- <exception caught here> ---
File "/Users/lthibault/.pyenv/versions/3.6.2/lib/python3.6/site-packages/twisted/application/app.py", line 464, in getApplication
application = service.loadApplication(filename, style, passphrase)
File "/Users/lthibault/.pyenv/versions/3.6.2/lib/python3.6/site-packages/twisted/application/service.py", line 416, in loadApplication
application = sob.loadValueFromFile(filename, 'application')
File "/Users/lthibault/.pyenv/versions/3.6.2/lib/python3.6/site-packages/twisted/persisted/sob.py", line 177, in loadValueFromFile
eval(codeObj, d, d)
File "test.tac", line 1, in <module>
from foo.bar import Bar
builtins.ModuleNotFoundError: No module named 'foo'

Failed to load application: No module named 'foo'

这是非常令人惊讶的,因为python test.tac不会产生错误。 为了进一步调试,我修改test.tac如下:

from sys import path
print(path)
from foo.bar import Bar

这表明运行python <filename>会将当前工作目录附加到路径,而运行twistd <filename>则不会。

我的问题是双重的:

  1. 这是一个错误,还是twistd有充分的理由这样做?
  2. twistd中解决此问题的推荐方法是什么. 我可以设置某种选项还是需要手动完成?

此代码段会将包含 tac 文件的目录添加到 python 路径

import os
import sys
sys.path.append(os.path.abspath(os.path.dirname(__file__)))
from foo.bar import Bar

Python 中没有规则说工作目录应该在路径上。相反,默认情况下,Python 将主脚本的目录放在路径上。当你运行python test.tac时,该目录是工作目录,但是当你运行 twistd 时,主 Python 脚本是 Twisted 的某些部分,它不在工作目录中。

由于我不熟悉 Twisted,因此我无法说您应该使用哪种代码组织和 Twisted 配置组合来使此导入工作。

相关内容

  • 没有找到相关文章

最新更新