使用setuptools / distutils在Python中构建C库时
$ python setup.py build
*.so/*.pyd
文件放在build/lib.win32-2.7
(或等效文件)中。
我想在我的测试套件中测试这些文件,但我不想硬编码build/lib*
路径。有没有人知道如何从distutils拉这条路,所以我可以sys.path.append(build_path)
- 或者有更好的方法来获取这些文件? (没有先安装它们)
您必须获得正在运行的平台以及您正在运行的python版本,然后自己组装名称。
要获得当前平台,请使用sysconfig.get_platform()
。要获取python版本,请使用sys.version_info
(特别是返回元组的前三个元素)。在我的系统(使用python 2.7.2的64位Linux)上,我得到:
>>> import sysconfig
>>> import sys
>>> sysconfig.get_platform()
linux-x86_64
>>> sys.version_info[:3]
(2, 7, 2)
lib目录的格式是“lib.platform-versionmajor.versionminor”(即只有2.7,而不是2.7.2)。您可以使用python的字符串格式化方法构造此字符串:
def distutils_dir_name(dname):
"""Returns the name of a distutils build directory"""
f = "{dirname}.{platform}-{version[0]}.{version[1]}"
return f.format(dirname=dname,
platform=sysconfig.get_platform(),
version=sys.version_info)
您可以使用它来生成任何distutils构建目录的名称:
>>> import os
>>> os.path.join('build', distutils_dir_name('lib'))
build/lib.linux-x86_64-2.7
我发现就地编译模块最适合测试目的。这样做只需使用
python setup.py build_ext --inplace
这将像往常一样编译临时目录中的所有辅助文件,但最终的.so文件将放在当前目录中。
这可以作为build_lib
命令类的build
成员,它与setup.py
的位置有关。当你没有真正运行setup
时,实现它是相当讨厌的:
import distutils.dist
import distutils.command.build
b = distutils.command.build.build(distutils.dist.Distribution())
b.finalize_options()
print(b.build_lib)
我个人会选择SethMMorton的解决方案,因为如果distutils
改变了构建目录位置,它可能很脆弱,我怀疑如果distutils
结构发生变化,玩这些技巧会更加脆弱。
@ SethMMorton的方法是要走的路。然而,使用来自get_platform()
的distutils.util
而不是sysconfig
可能更安全(即使我不知道他的版本不起作用的平台)。
这是它在distutils-library中的使用方式:
import os
import sys
from distutils.util import get_platform
def get_distutils_lib_path():
PLAT_SPEC = "%s-%d.%d" % (get_platform(), *sys.version_info[:2])
return os.path.join("build", "lib.%s" % PLAT_SPEC)