导入Python命名空间包的本地测试版本

时间:2015-09-12 17:41:59

标签: python python-2.7 python-import anaconda namespace-package

我想知道如何导入Python 2.7命名空间包的本地测试版本。在此示例中,包Ska.engarchiveSka根目录下的命名空间包。 (这种结构是由遗产强加给我的)。

此示例显示即使将sys.path设置为以本地目录开头,也会导入已安装的软件包版本,而不是本地版本。

Python 2.7.9 |Continuum Analytics, Inc.| (default, Apr 14 2015, 12:54:25)
... 
In [1]: import sys
In [2]: sys.path.insert(0, '.')
In [3]: import Ska.engarchive.fetch_eng as fetch
In [4]: fetch.__file__
Out[4]: '/proj/sot/ska/arch/x86_64-linux_CentOS-5/lib/python2.7/site-packages/Ska.engarchive-0.36.2-py2.7.egg/Ska/engarchive/fetch_eng.pyc'

我认为问题与在Python 2中实现命名空间包的方式有关,无论如何,命名空间路径始终位于列表的前面。但也许有一些解决方法?我花了一些时间挖掘site软件包文档,但也许我没有看到正确的事情。

以上示例使用的是Anaconda Python发行版。有趣的是,如果我使用ActiveState中的旧版Python,该示例具有导入本地包的所需结果:

Python 2.7.1 (r271:86832, Feb  7 2011, 11:30:54) 

In [1]: import sys
In [2]: sys.path.insert(0, '.')
In [3]: import Ska.engarchive.fetch_eng as fetch
In [4]: fetch.__file__
Out[4]: './Ska/engarchive/fetch_eng.pyc'

非常感谢任何帮助!

1 个答案:

答案 0 :(得分:2)

我能够使用namespace packages的setuptools方法复制此行为(但不是标准的pkgutil method)。 Setuptools将导入已安装的软件包,而 pkgutil 将导入本地软件包。 Setuptools似乎从您期望的方向向后加载__path__(首先安装,本地安装第二个)。例如,(参见nstest定义的底部示例)

>>> import nstest
>>> nstest.__path__
['/home/caleb/.local/lib/python2.7/site-packages/nstest.foo-0.0.0-py2.7.egg/nstest', 'nstest']

解决这个问题的方法是将__path__前面的本地包添加到最右边的命名空间包中。如,

>>> import nstest
>>> nstest.__path__.insert(0, 'nstest')
>>> from nstest import foo
>>> foo.__file__
'nstest/foo/__init__.py'

由于您在问题中说Ska.engarchive是命名空间包,因此您需要在解释器中执行以下操作:

>>> import Ska.engarchive
>>> Ska.engarchive.__path__.insert(0, 'Ska/engarchive')
>>> import Ska.engarchive.fetch_eng as fetch
>>> fetch.__file__
'Ska/engarchive/fetch_eng.pyc' # This should be outputted

使用setuptools的 pkg_resources

进行命名空间测试

目录结构:

nstest.foo/
├─ setup.py
└─ nstest/
   ├─ __init__.py
   └─ foo/
      └─ __init__.py

nstest.foo/setup.py

from setuptools import setup, find_packages
setup(name='nstest.foo', packages=find_packages())

nstest.foo/nstest / __初始化__吡啶

__import__('pkg_resources').declare_namespace(__name__)

nstest.foo/nstest/foo / __初始化__吡啶

# empty

安装:

$ python2 setup.py build
$ python2 setup.py install --user

测试:

$ python2
>>> from nstest import foo
>>> foo.__file__
'/home/caleb/.local/lib/python2.7/site-packages/nstest.foo-0.0.0-py2.7.egg/nstest/foo/__init__.pyc'