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

我在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

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

现在您可以说:有什么问题?使用virtualenv并完成它.但是,需要通过pip额外安装virtualenv,这需要我并不总是拥有的管理员权限. 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年的bug report仍然没有修复).

#!/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'))
上一篇:virtualenv虚拟环境使用及介绍


下一篇:我如何诊断ImportError:在virtualenv中加载本机Windows python模块时,DLL加载失败?