为什么"pyvenv"不安装"python config"

Why `pyvenv` does not install `python-config`?

本文关键字:quot python config 安装 pyvenv 为什么 不安      更新时间:2023-10-16

我在MacOS(10.11)下遇到过这种情况,但在各种Linux下也遇到过同样的问题。我安装了"官方"Python3包,它进入了/Library/Frameworks/Python.framework/Versions/3.4。(注意:下面的例子使用Python 3.4,但问题在3.5中仍然存在。如果问题在3.6中得到解决,由于缺乏管理权限,我无法访问具有Python 3.6的机器。)

我需要虚拟环境,我需要python-config脚本来确定Python3使用了哪些库,因为我的项目结合了Python和C++代码。

如果我用virtualenv设置虚拟环境,一切都很好:

$ which virtualenv
/Library/Frameworks/Python.framework/Versions/3.4/bin/virtualenv
$ virtualenv --python=$(which python3) vienv
Running virtualenv with interpreter /Library/Frameworks/Python.framework/Versions/3.4/bin/python3
Using base prefix '/Library/Frameworks/Python.framework/Versions/3.4'
[...blabla...]
Installing setuptools, pip, wheel...done.
$ source vienv/bin/activate
(vienv) $ which python-config
/Users/XXXXX/DEV/STANDALONE/misc/python/vienv/bin/python-config
(vienv) $ python-config --libs
-lpython3.4m -ldl -framework CoreFoundation

然而,pyvenv忘记在虚拟环境中设置python-config

$which pyvenv
/Library/Frameworks/Python.framework/Versions/3.4/bin/pyvenv
$ pyvenv pe
$ source pe/bin/activate
(pe) $ which python-config
/usr/bin/python-config   # !!! Here's the problem !!!
(pe) $ python-config --libs
-lpython2.7 -ldl -framework CoreFoundation

换句话说,即使我激活了虚拟环境,系统默认的Python2python-config仍保留在我的PATH中。

现在你可以说:怎么了?使用virtualenv并完成它。然而,virtualenv需要通过pip额外安装,这需要我并不总是拥有的管理权限。pyvenv,OTOH,与Python3一起出现,或者至少我是这么理解的。

您还可以说:为什么不使用pip在虚拟环境中安装python-config呢?原因如下:

(pe) $ pip install python-config
Requirement already satisfied (use --upgrade to upgrade): python-config in ./pe/lib/python3.4/site-packages
Cleaning up...

是的,在那里,但脚本本身没有安装到虚拟环境的bin子目录中。

摘要:我想配置我的项目,使其只能使用Python3标准模块/工具进行安装,而不依赖于virtualenv等额外的东西。我不想纠缠系统管理员:-)

问题:有没有一个解决方法可以让pyvenv正确安装python-config?或者:如果我将C++代码链接到虚拟环境中的特定Python3安装,是否有其他方法可以确定我应该使用哪些头和库?

好吧,一年后是时候回答我自己的问题了:-)

下面是由virtualenv安装到${VENV}/bin中的python-config脚本。如果你使用python3 -m venv ${VENV},那么只需手动将其复制到这个位置,直到这个问题得到修复(注意,据我所知,有一份2011年的错误报告仍然没有修复)。

#!/usr/bin/env python3
"""
This python-config script was taken from a virtual environment
created by `virtualenv`.
The only change is the hash-bang line.
The user shall copy this to ${VENV}/bin during setup.
:author: unknown + AA
:date: 2018-02-23
"""
import sys
import getopt
import sysconfig
valid_opts = ['prefix', 'exec-prefix', 'includes', 'libs', 'cflags',
'ldflags', 'help']
if sys.version_info >= (3, 2):
valid_opts.insert(-1, 'extension-suffix')
valid_opts.append('abiflags')
if sys.version_info >= (3, 3):
valid_opts.append('configdir')

def exit_with_usage(code=1):
sys.stderr.write("Usage: {0} [{1}]n".format(
sys.argv[0], '|'.join('--'+opt for opt in valid_opts)))
sys.exit(code)
try:
opts, args = getopt.getopt(sys.argv[1:], '', valid_opts)
except getopt.error:
exit_with_usage()
if not opts:
exit_with_usage()
pyver = sysconfig.get_config_var('VERSION')
getvar = sysconfig.get_config_var
opt_flags = [flag for (flag, val) in opts]
if '--help' in opt_flags:
exit_with_usage(code=0)
for opt in opt_flags:
if opt == '--prefix':
print(sysconfig.get_config_var('prefix'))
elif opt == '--exec-prefix':
print(sysconfig.get_config_var('exec_prefix'))
elif opt in ('--includes', '--cflags'):
flags = ['-I' + sysconfig.get_path('include'),
'-I' + sysconfig.get_path('platinclude')]
if opt == '--cflags':
flags.extend(getvar('CFLAGS').split())
print(' '.join(flags))
elif opt in ('--libs', '--ldflags'):
abiflags = getattr(sys, 'abiflags', '')
libs = ['-lpython' + pyver + abiflags]
libs += getvar('LIBS').split()
libs += getvar('SYSLIBS').split()
# add the prefix/lib/pythonX.Y/config dir, but only if there is no
# shared library in prefix/lib/.
if opt == '--ldflags':
if not getvar('Py_ENABLE_SHARED'):
libs.insert(0, '-L' + getvar('LIBPL'))
if not getvar('PYTHONFRAMEWORK'):
libs.extend(getvar('LINKFORSHARED').split())
print(' '.join(libs))
elif opt == '--extension-suffix':
ext_suffix = sysconfig.get_config_var('EXT_SUFFIX')
if ext_suffix is None:
ext_suffix = sysconfig.get_config_var('SO')
print(ext_suffix)
elif opt == '--abiflags':
if not getattr(sys, 'abiflags', None):
exit_with_usage()
print(sys.abiflags)
elif opt == '--configdir':
print(sysconfig.get_config_var('LIBPL'))