uvloop/setup.py

325 lines
10 KiB
Python
Raw Normal View History

import sys
vi = sys.version_info
if vi < (3, 7):
raise RuntimeError('uvloop requires Python 3.7 or greater')
if sys.platform in ('win32', 'cygwin', 'cli'):
raise RuntimeError('uvloop does not support Windows at the moment')
2015-11-01 17:00:43 +00:00
import os
import os.path
import pathlib
import platform
import re
import shutil
2015-11-01 17:00:43 +00:00
import subprocess
import sys
from setuptools import setup, Extension
2021-09-05 10:19:30 +00:00
from setuptools.command.build_ext import build_ext
from setuptools.command.sdist import sdist
CYTHON_DEPENDENCY = 'Cython(>=0.29.24,<0.30.0)'
# Minimal dependencies required to test uvloop.
TEST_DEPENDENCIES = [
# pycodestyle is a dependency of flake8, but it must be frozen because
# their combination breaks too often
# (example breakage: https://gitlab.com/pycqa/flake8/issues/427)
'aiohttp',
2021-08-07 01:02:12 +00:00
'flake8~=3.9.2',
'psutil',
2021-08-07 01:02:12 +00:00
'pycodestyle~=2.7.0',
'pyOpenSSL~=19.0.0',
'mypy>=0.800',
CYTHON_DEPENDENCY,
]
# Dependencies required to build documentation.
DOC_DEPENDENCIES = [
2021-08-07 01:02:12 +00:00
'Sphinx~=4.1.2',
'sphinxcontrib-asyncio~=0.3.0',
'sphinx_rtd_theme~=0.5.2',
]
EXTRA_DEPENDENCIES = {
'docs': DOC_DEPENDENCIES,
'test': TEST_DEPENDENCIES,
# Dependencies required to develop uvloop.
'dev': [
CYTHON_DEPENDENCY,
'pytest>=3.6.0',
] + DOC_DEPENDENCIES + TEST_DEPENDENCIES
}
MACHINE = platform.machine()
MODULES_CFLAGS = [os.getenv('UVLOOP_OPT_CFLAGS', '-O2')]
_ROOT = pathlib.Path(__file__).parent
LIBUV_DIR = str(_ROOT / 'vendor' / 'libuv')
LIBUV_BUILD_DIR = str(_ROOT / 'build' / 'libuv-{}'.format(MACHINE))
2015-11-01 17:00:43 +00:00
def _libuv_build_env():
env = os.environ.copy()
cur_cflags = env.get('CFLAGS', '')
2018-06-21 15:43:38 +00:00
if not re.search(r'-O\d', cur_cflags):
cur_cflags += ' -O2'
env['CFLAGS'] = (cur_cflags + ' -fPIC ' + env.get('ARCHFLAGS', ''))
return env
def _libuv_autogen(env):
if os.path.exists(os.path.join(LIBUV_DIR, 'configure')):
# No need to use autogen, the configure script is there.
return
if not os.path.exists(os.path.join(LIBUV_DIR, 'autogen.sh')):
raise RuntimeError(
'the libuv submodule has not been checked out; '
'try running "git submodule init; git submodule update"')
subprocess.run(
['/bin/sh', 'autogen.sh'], cwd=LIBUV_DIR, env=env, check=True)
class uvloop_sdist(sdist):
def run(self):
# Make sure sdist archive contains configure
# to avoid the dependency on autotools.
_libuv_autogen(_libuv_build_env())
super().run()
class uvloop_build_ext(build_ext):
user_options = build_ext.user_options + [
('cython-always', None,
'run cythonize() even if .c files are present'),
('cython-annotate', None,
'Produce a colorized HTML version of the Cython source.'),
('cython-directives=', None,
'Cythion compiler directives'),
('use-system-libuv', None,
'Use the system provided libuv, instead of the bundled one'),
]
boolean_options = build_ext.boolean_options + [
'cython-always',
'cython-annotate',
'use-system-libuv',
]
def initialize_options(self):
# initialize_options() may be called multiple times on the
# same command object, so make sure not to override previously
# set options.
if getattr(self, '_initialized', False):
return
super().initialize_options()
self.use_system_libuv = False
self.cython_always = False
self.cython_annotate = None
self.cython_directives = None
def finalize_options(self):
# finalize_options() may be called multiple times on the
# same command object, so make sure not to override previously
# set options.
if getattr(self, '_initialized', False):
return
need_cythonize = self.cython_always
cfiles = {}
for extension in self.distribution.ext_modules:
for i, sfile in enumerate(extension.sources):
if sfile.endswith('.pyx'):
prefix, ext = os.path.splitext(sfile)
cfile = prefix + '.c'
if os.path.exists(cfile) and not self.cython_always:
extension.sources[i] = cfile
else:
if os.path.exists(cfile):
cfiles[cfile] = os.path.getmtime(cfile)
else:
cfiles[cfile] = 0
need_cythonize = True
if need_cythonize:
import pkg_resources
# Double check Cython presence in case setup_requires
# didn't go into effect (most likely because someone
# imported Cython before setup_requires injected the
# correct egg into sys.path.
try:
import Cython
except ImportError:
raise RuntimeError(
'please install {} to compile uvloop from source'.format(
CYTHON_DEPENDENCY))
cython_dep = pkg_resources.Requirement.parse(CYTHON_DEPENDENCY)
if Cython.__version__ not in cython_dep:
raise RuntimeError(
'uvloop requires {}, got Cython=={}'.format(
CYTHON_DEPENDENCY, Cython.__version__
))
from Cython.Build import cythonize
directives = {}
if self.cython_directives:
for directive in self.cython_directives.split(','):
k, _, v = directive.partition('=')
if v.lower() == 'false':
v = False
if v.lower() == 'true':
v = True
directives[k] = v
self.distribution.ext_modules[:] = cythonize(
self.distribution.ext_modules,
compiler_directives=directives,
annotate=self.cython_annotate)
super().finalize_options()
self._initialized = True
2015-11-01 17:00:43 +00:00
def build_libuv(self):
env = _libuv_build_env()
# Make sure configure and friends are present in case
# we are building from a git checkout.
_libuv_autogen(env)
2015-11-01 17:00:43 +00:00
# Copy the libuv tree to build/ so that its build
# products don't pollute sdist accidentally.
if os.path.exists(LIBUV_BUILD_DIR):
shutil.rmtree(LIBUV_BUILD_DIR)
shutil.copytree(LIBUV_DIR, LIBUV_BUILD_DIR)
2016-05-05 17:06:10 +00:00
# Sometimes pip fails to preserve the timestamps correctly,
# in which case, make will try to run autotools again.
subprocess.run(
['touch', 'configure.ac', 'aclocal.m4', 'configure',
'Makefile.am', 'Makefile.in'],
cwd=LIBUV_BUILD_DIR, env=env, check=True)
if 'LIBUV_CONFIGURE_HOST' in env:
cmd = ['./configure', '--host=' + env['LIBUV_CONFIGURE_HOST']]
else:
cmd = ['./configure']
subprocess.run(
cmd,
cwd=LIBUV_BUILD_DIR, env=env, check=True)
j_flag = '-j{}'.format(os.cpu_count() or 1)
c_flag = "CFLAGS={}".format(env['CFLAGS'])
subprocess.run(
['make', j_flag, c_flag],
cwd=LIBUV_BUILD_DIR, env=env, check=True)
2015-11-01 17:00:43 +00:00
def build_extensions(self):
if self.use_system_libuv:
self.compiler.add_library('uv')
if sys.platform == 'darwin' and \
os.path.exists('/opt/local/include'):
# Support macports on Mac OS X.
self.compiler.add_include_dir('/opt/local/include')
else:
libuv_lib = os.path.join(LIBUV_BUILD_DIR, '.libs', 'libuv.a')
if not os.path.exists(libuv_lib):
self.build_libuv()
if not os.path.exists(libuv_lib):
raise RuntimeError('failed to build libuv')
self.extensions[-1].extra_objects.extend([libuv_lib])
self.compiler.add_include_dir(os.path.join(LIBUV_DIR, 'include'))
2015-11-01 17:00:43 +00:00
if sys.platform.startswith('linux'):
self.compiler.add_library('rt')
elif sys.platform.startswith(('freebsd', 'dragonfly')):
2015-11-01 17:00:43 +00:00
self.compiler.add_library('kvm')
2016-08-23 11:51:26 +00:00
elif sys.platform.startswith('sunos'):
self.compiler.add_library('kstat')
2017-11-26 06:32:14 +00:00
self.compiler.add_library('pthread')
2015-11-01 17:00:43 +00:00
super().build_extensions()
with open(str(_ROOT / 'README.rst')) as f:
readme = f.read()
with open(str(_ROOT / 'uvloop' / '_version.py')) as f:
for line in f:
if line.startswith('__version__ ='):
_, _, version = line.partition('=')
VERSION = version.strip(" \n'\"")
break
else:
raise RuntimeError(
2020-12-06 17:46:20 +00:00
'unable to read the version from uvloop/_version.py')
setup_requires = []
if not (_ROOT / 'uvloop' / 'loop.c').exists() or '--cython-always' in sys.argv:
# No Cython output, require Cython to build.
setup_requires.append(CYTHON_DEPENDENCY)
2015-11-01 17:00:43 +00:00
setup(
name='uvloop',
2016-04-14 04:45:01 +00:00
description='Fast implementation of asyncio event loop on top of libuv',
long_description=readme,
2016-05-07 01:25:15 +00:00
url='http://github.com/MagicStack/uvloop',
license='MIT',
author='Yury Selivanov',
author_email='yury@magic.io',
platforms=['macOS', 'POSIX'],
version=VERSION,
2015-11-01 17:00:43 +00:00
packages=['uvloop'],
cmdclass={
'sdist': uvloop_sdist,
'build_ext': uvloop_build_ext
},
2015-11-01 17:00:43 +00:00
ext_modules=[
Extension(
2015-11-02 20:11:19 +00:00
"uvloop.loop",
2016-04-14 04:45:01 +00:00
sources=[
"uvloop/loop.pyx",
2016-05-18 23:53:36 +00:00
],
extra_compile_args=MODULES_CFLAGS
2015-11-02 20:11:19 +00:00
),
2015-11-01 17:00:43 +00:00
],
2016-04-14 04:45:01 +00:00
classifiers=[
'Development Status :: 5 - Production/Stable',
2018-04-17 20:11:34 +00:00
'Framework :: AsyncIO',
2016-07-16 15:19:42 +00:00
'Programming Language :: Python :: 3 :: Only',
2018-05-30 20:51:40 +00:00
'Programming Language :: Python :: 3.7',
'Programming Language :: Python :: 3.8',
'Programming Language :: Python :: 3.9',
2021-08-07 01:02:12 +00:00
'Programming Language :: Python :: 3.10',
'License :: OSI Approved :: Apache Software License',
2016-04-14 04:45:01 +00:00
'License :: OSI Approved :: MIT License',
'Intended Audience :: Developers',
],
2016-05-11 16:17:10 +00:00
include_package_data=True,
extras_require=EXTRA_DEPENDENCIES,
2021-02-18 20:57:04 +00:00
setup_requires=setup_requires,
python_requires='>=3.7',
2015-11-01 17:00:43 +00:00
)