为什么`pyvenv`不安装`python-config`?

Lar*_*dua 6 c++ virtualenv python-3.x python-config pyvenv

我在 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
Run Code Online (Sandbox Code Playgroud)

但是,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
Run Code Online (Sandbox Code Playgroud)

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

现在你可以说:有什么问题吗?使用virtualenv并完成它。但是,virtualenv需要额外安装pip,这需要我并不总是拥有的管理员权限。 pyvenv,OTOH,随 Python3 一起提供,或者至少这是我的理解。

你也可以说:你为什么不直接python-config在你的虚拟环境中使用pip? 原因如下:

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

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

总结:我想配置我的项目,以便它只能使用 Python3 标准模块/工具进行安装,并且它不依赖于virtualenv. 我不想纠缠系统管理员:-)

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

Lar*_*dua 9

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

下面python-config是通过virtualenvinto安装的脚本${VENV}/bin。如果您使用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'))
Run Code Online (Sandbox Code Playgroud)

  • 我不敢相信 virtualenv 不会创建新文件...花了我 1 小时来调试。 (2认同)