Я думаю, вам придется внести некоторые изменения как в ваш setup.py
, так и в один файл __init__
в ваш модуль.
Пусть говорят имя пакета будет «модуль» и у вас есть функциональность, sub
, для которых у вас есть чистый код питона в sub
вложенной и эквивалентный код C в c_sub
вложенной. Например, в вашей setup.py
:
import logging
from setuptools.extension import Extension
from setuptools.command.build_ext import build_ext
from distutils.errors import CCompilerError, DistutilsExecError, DistutilsPlatformError
logging.basicConfig()
log = logging.getLogger(__file__)
ext_errors = (CCompilerError, DistutilsExecError, DistutilsPlatformError, IOError)
class BuildFailed(Exception):
pass
def construct_build_ext(build_ext):
class WrappedBuildExt(build_ext):
# This class allows C extension building to fail.
def run(self):
try:
build_ext.run(self)
except DistutilsPlatformError as x:
raise BuildFailed(x)
def build_extension(self, ext):
try:
build_ext.build_extension(self, ext)
except ext_errors as x:
raise BuildFailed(x)
return WrappedBuildExt
setup_args = {'name': 'module', 'license': 'BSD', 'author': 'xxx',
'packages': ['module', 'module.sub', 'module.c_sub'],
'cmdclass': {}
}
ext_modules = [Extension("module.c_sub._sub", ["module/c_sub/_sub.c"])]
cmd_classes = setup_args.setdefault('cmdclass', {})
try:
# try building with c code :
setup_args['cmdclass']['build_ext'] = construct_build_ext(build_ext)
setup(ext_modules=ext_modules, **setup_args)
except BuildFailed as ex:
log.warn(ex)
log.warn("The C extension could not be compiled")
## Retry to install the module without C extensions :
# Remove any previously defined build_ext command class.
if 'build_ext' in setup_args['cmdclass']:
del setup_args['cmdclass']['build_ext']
if 'build_ext' in cmd_classes:
del cmd_classes['build_ext']
# If this new 'setup' call don't fail, the module
# will be successfully installed, without the C extension :
setup(**setup_args)
log.info("Plain-Python installation succeeded.")
Теперь вам нужно будет включать в себя что-то вроде этого в файле __init__.py
(или в любом месте, соответствующем в вашем случае):
try:
from .c_sub import *
except ImportError:
from .sub import *
Таким образом, C версия будет использоваться, если она была построена, в противном случае используется простая версия python. Предполагается, что sub
и c_sub
предоставят тот же API.
Вы можете найти example of setup file таким образом в пакете Shapely
. Фактически большая часть кода, который я опубликовал, была скопирована (функция construct_build_ext
) или адаптирована (строки после) из этого файла.
Я не вижу, как это применимо. Они пытаются импортировать cython в качестве модуля python и возвращаться к C-модулю, если импорт не выполняется. Как вы предлагаете мне попробовать импортировать системный компилятор C в python? – ARF
Действительно, я немного подпрыгнул до выводов. –