Почему `pyvenv` не устанавливает`python-config`?

Я сталкивался с этим под MacOS (10.11), но видел такую ​​же проблему и в разных Linux. Я установил "официальный" пакет Python3, он идет в /Library/Frameworks/Python.framework/Versions/3.4, (Примечание: в примерах ниже используется Python 3.4, но проблема сохраняется и с 3.5. У меня нет доступа к машине с Python 3.6 из-за отсутствия привилегий администратора, если проблема будет решена в 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 и покончим с этим. Тем не мение, 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...

Да, пакет есть, но сам скрипт не установлен в bin подкаталог виртуальной среды.

Резюме: я хотел бы настроить свой проект так, чтобы он мог быть установлен только с использованием стандартных модулей / инструментов Python3, и он не зависел от дополнительных вещей, таких как virtualenv, И я не хочу приставать к сисадминам:-)

Вопрос: есть ли обходной путь, чтобы получить pyvenv устанавливать python-config должным образом? Или: есть ли другой способ выяснить, какие заголовки и библиотеки мне следует использовать, если я свяжу свой код C++ с конкретной установкой Python3 в виртуальной среде?

1 ответ

Что ж, через год пришло время ответить на мой вопрос:-)

Здесь следует python-config скрипт, который устанавливается virtualenv в ${VENV}/bin, Если вы используете python3 -m venv ${VENV}, а затем просто скопируйте его в это место вручную, пока эта проблема не будет устранена (NB, насколько я могу судить, отчет об ошибке за 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'))
Другие вопросы по тегам