python setuptools install_requires игнорируется при переопределении cmdclass
У меня есть setup.py
это выглядит так:
from setuptools import setup
from subprocess import call
from setuptools.command.install import install
class MyInstall(install):
def run(self):
call(["pip install -r requirements.txt --no-clean"], shell=True)
install.run(self)
setup(
author='Attila Zseder',
version='0.1',
name='entity_extractor',
packages=['...'],
install_requires=['DAWG', 'mrjob', 'cchardet'],
package_dir={'': 'modules'},
scripts=['...'],
cmdclass={'install': MyInstall},
)
я нуждаюсь MyInstall
потому что я хочу установить некоторые библиотеки из GitHub, и я не хотел использовать dependency_links
вариант, потому что это не рекомендуется (например, здесь), поэтому я могу сделать это с помощью require.txt.
Когда я устанавливаю этот пакет с pip
все работает нормально, но по некоторым причинам я должен решить это так, чтобы он также работал с чистым python setup.py install
, И это не так.
При переопределении cmdclass
в setup()
с моим собственным классом, install_requires
кажется, игнорируется. Как только я закомментирую эту строку, эти пакеты устанавливаются.
Я знаю, что install_requires не поддерживается, например, в distutils (если я хорошо помню), но он есть в setuptools. А потом cmdclass
не будет иметь никакого влияния на install_requires
,
Я часами гуглял эту проблему, нашел много похожих ответов на stackru, но не для этой конкретной проблемы.
С размещением каждого необходимого пакета в файле require.txt все работает нормально, но я хотел бы понять, почему это происходит. Спасибо!
3 ответа
Та же самая проблема только что случилась со мной. Как-то кажется, что что-то запускает setuptools, чтобы сделать "установку в старом стиле" с distutils
что действительно не поддерживает install_requires
,
Вы вызываете install.run(self), который вызывает run (self) в setuptools/setuptools/command/install.py, строка 51-74.
def run(self):
# Explicit request for old-style install? Just do it
if self.old_and_unmanageable or self.single_version_externally_managed:
return _install.run(self)
# Attempt to detect whether we were called from setup() or by another
# command. If we were called by setup(), our caller will be the
# 'run_command' method in 'distutils.dist', and *its* caller will be
# the 'run_commands' method. If we were called any other way, our
# immediate caller *might* be 'run_command', but it won't have been
# called by 'run_commands'. This is slightly kludgy, but seems to
# work.
#
caller = sys._getframe(2)
caller_module = caller.f_globals.get('__name__','')
caller_name = caller.f_code.co_name
if caller_module != 'distutils.dist' or caller_name!='run_commands':
# We weren't called from the command line or setup(), so we
# should run in backward-compatibility mode to support bdist_*
# commands.
_install.run(self)
else:
self.do_egg_install()
Я не уверен, предназначено ли это поведение, но заменяет
install.run(self)
с
install.do_egg_install()
должен решить вашу проблему. По крайней мере, это работает для меня, но я также был бы признателен за более подробный ответ. Спасибо!
Согласно /questions/1584745/zapusk-polzovatelskih-sborok-setuptools-vo-vremya-ustanovki/1584756#1584756 более правильным способом для этого может быть переопределение bdist_egg
команда.
Вы можете попробовать:
from setuptools.command.bdist_egg import bdist_egg as _bdist_egg
class bdist_egg(_bdist_egg):
def run(self):
call(["pip install -r requirements.txt --no-clean"], shell=True)
_bdist_egg.run(self)
...
setup(...
cmdclass={'bdist_egg': bdist_egg}, # override bdist_egg
)
Это сработало для меня и install_require
больше не игнорируется. Тем не менее, я до сих пор не понимаю, почему большинство людей, кажется, игнорируют cmdclass install
и не жалуются на install_require
Быть игнорированным.
Я знаю, что это старый вопрос, но я столкнулся с аналогичной проблемой. Решение, которое я нашел, решает эту проблему для меня очень тонко: install
класс, в котором вы сидите cmd_class
должен быть физически назван install
, Смотрите этот ответ по связанной проблеме.
Обратите внимание, что я использую имя класса install для моего производного класса, потому что именно это будет использовать python setup.py --help-команды.
Вы также должны использовать self.execute(_func_name, (), msg="msg")
в вашем post_install вместо прямого вызова функции
Так что реализация чего-то подобного должна заставить вас избегать do_egg_install
Обходной путь, реализованный выше KEgg.
from setuptools.command.install import install as _install
...
def _post_install():
#code here
class install(_install):
def run(self):
_install.run(self)
self.execute(_post_install, (), msg="message here")