我将要破解Python导入系统。假设我们有以下目录结构:
.
├── main
│ ├── main.py
│ └── parent
│ └── __init__.py
└── pkg1
├── __init__.py
├── sub
│ ├── __init__.py
│ └── import_global.py
└── success.py
启动脚本将是main.py
,因此应该有一个最顶层的模块parent
。现在,我想模拟一个子包,它的全名是parent.intermediate.pkg1
,它确实引用了pkg1
目录。在
实际上不存在intermediate
模块,但是,我确实需要模拟一个模块(在我的实际项目中,这个中间模块的名称将动态生成)。所以我决定使用Python导入钩子。在
首先,让我介绍一下pkg1
的内容。在
包装1/分包/进口_全局.py公司名称:
^{pr2}$包装1/成功.py公司名称:
Value = 'Success'
和(部分主.py),我做了一些测试用例:
class MainTestCase(unittest.TestCase):
def test_success(self):
from parent.intermediate.pkg1 import success
self.assertEqual(success.Value, "Success")
def test_import_global(self):
from parent.intermediate.pkg1.sub import import_global
self.assertEqual(import_global.Value, 3)
def test_not_found(self):
def F():
from parent.intermediate.pkg1 import not_found
self.assertRaises(ImportError, F)
unittest.main()
所有的__init__.py
都是空的。现在它将实现导入钩子。我起草了两个版本,每个版本都有一些问题。在
第一版:
class PkgLoader(object):
def install(self):
sys.meta_path[:] = [x for x in sys.meta_path if self != x] + [self]
def find_module(self, fullname, path=None):
if fullname.startswith('parent.'):
return self
def load_module(self, fullname):
if fullname in sys.modules:
return sys.modules[fullname]
parts = fullname.split('.')[1:]
path = os.path.join(os.path.dirname(__file__), '..')
# intermediate module
m = None
ns = 'parent.intermediate'
if ns in sys.modules:
m = sys.modules[ns]
elif parts[0] == 'intermediate':
m = imp.new_module(ns)
m.__name__ = ns
m.__path__ = [ns]
m.__package__ = '.'.join(ns.rsplit('.', 1)[:-1])
else:
raise ImportError("Module %s not found." % fullname)
# submodules
for p in parts[1:]:
ns = '%s.%s' % (ns, p)
fp, filename, options = imp.find_module(p, [path])
if ns in sys.modules:
m = sys.modules[ns]
else:
m = imp.load_module(ns, fp, filename, options)
sys.modules[ns] = m
path = filename
return m
loader = PkgLoader()
loader.install()
如果test_import_global
失败:
E..
======================================================================
ERROR: test_import_global (__main__.MainTestCase)
----------------------------------------------------------------------
Traceback (most recent call last):
File "main.py", line 54, in test_import_global
from parent.intermediate.pkg1.sub import import_global
File "main.py", line 39, in load_module
m = imp.load_module(ns, fp, filename, options)
File "../pkg1/sub/import_global.py", line 1, in <module>
from operator import add
File "main.py", line 35, in load_module
fp, filename, options = imp.find_module(p, [path])
ImportError: No module named operator
----------------------------------------------------------------------
Ran 3 tests in 0.005s
FAILED (errors=1)
现在对于第二个版本,我修改了load_module
:
def load_module(self, fullname):
if fullname in sys.modules:
return sys.modules[fullname]
parts = fullname.split('.')[1:]
path = os.path.join(os.path.dirname(__file__), '..')
# intermediate module
m = None
ns = 'parent.intermediate'
if ns in sys.modules:
m = sys.modules[ns]
elif parts[0] == 'intermediate':
m = imp.new_module(ns)
m.__name__ = ns
m.__path__ = [ns]
m.__package__ = '.'.join(ns.rsplit('.', 1)[:-1])
else:
raise ImportError("Module %s not found." % fullname)
# submodules
for p in parts[1:]:
ns = '%s.%s' % (ns, p)
# ======> The modification starts here <======
try:
fp, filename, options = imp.find_module(p, [path])
except ImportError:
return None
# ======> The modification ends here <======
if ns in sys.modules:
m = sys.modules[ns]
else:
m = imp.load_module(ns, fp, filename, options)
sys.modules[ns] = m
path = filename
return m
如果test_not_found
失败:
.F.
======================================================================
FAIL: test_not_found (__main__.MainTestCase)
----------------------------------------------------------------------
Traceback (most recent call last):
File "main.py", line 65, in test_not_found
self.assertRaises(ImportError, F)
AssertionError: ImportError not raised
----------------------------------------------------------------------
Ran 3 tests in 0.004s
FAILED (failures=1)
所以问题现在很清楚了:如何实现import钩子,以便这三个测试用例都能通过?在
哦,我有一个解决方案,尽管我的实际项目可能需要更多的测试用例。基本观点是在
find_module
阶段执行imp.find_module
,而不是load_module
阶段,这样可以避免系统使用我们定制的加载程序来加载不存在的模块。在解决方案如下:
请随意评论我的解决方案,如果你们发现任何潜在的错误,请随时通知我。在
您可以在运行时创建模块,也可以修改
sys.modules
字典。在所以,如果你有一个目录结构,比如:
当然,你也可以这样做:
^{pr2}$但是,如果您想“假装”
sub
实际上是另一个包中的一个子包,则可以执行以下操作:在我的测试代码中,
sub
的__init__.py
只包含:如果你运行
main.py
,你会得到:我认为这样的方法比使用导入钩子要容易得多。在
相关问题 更多 >
编程相关推荐