pyzmq/setup.py

1376 lines
46 KiB
Python
Executable File

#!/usr/bin/env python
#
# -----------------------------------------------------------------------------
# Copyright (C) PyZMQ Developers
# Distributed under the terms of the Modified BSD License.
#
# The `configure` subcommand is copied and adapted from h5py
# h5py source used under the New BSD license
#
# h5py: <http://code.google.com/p/h5py/>
#
# The code to bundle libzmq as an Extension is from pyzmq-static
# pyzmq-static source used under the New BSD license
#
# pyzmq-static: <https://github.com/brandon-rhodes/pyzmq-static>
# -----------------------------------------------------------------------------
import copy
import errno
import os
import platform
import shutil
import subprocess
import sys
import time
from contextlib import contextmanager
from glob import glob
from os.path import basename
from os.path import join as pjoin
from subprocess import PIPE, CalledProcessError, Popen, check_call
from sysconfig import get_config_var
from traceback import print_exc
from packaging.version import Version as V
from setuptools import Command, setup
from setuptools.command.bdist_egg import bdist_egg
from setuptools.command.build_ext import build_ext
from setuptools.command.sdist import sdist
from setuptools.extension import Extension
# local script imports:
sys.path.insert(0, os.path.dirname(__file__))
from buildutils import (
bundled_version,
compile_and_forget,
config_from_prefix,
customize_mingw,
detect_zmq,
discover_settings,
fatal,
fetch_libzmq,
fetch_libzmq_dll,
info,
line,
localpath,
locate_vcredist_dir,
merge,
patch_lib_paths,
save_config,
stage_platform_hpp,
v_str,
warn,
)
# -----------------------------------------------------------------------------
# Flags
# -----------------------------------------------------------------------------
# name of the libzmq library - can be changed by --libzmq <name>
libzmq_name = "libzmq"
doing_bdist = any(arg.startswith("bdist") for arg in sys.argv[1:])
pypy = platform.python_implementation() == 'PyPy'
# reference points for zmq compatibility
min_legacy_zmq = (2, 1, 4)
min_good_zmq = (3, 2)
target_zmq = bundled_version
dev_zmq = (target_zmq[0], target_zmq[1] + 1, 0)
# set dylib ext:
if sys.platform.startswith('win'):
lib_ext = '.dll'
elif sys.platform == 'darwin':
lib_ext = '.dylib'
else:
lib_ext = '.so'
# allow `--zmq=foo` to be passed at any point,
# but always assign it to configure
configure_idx = -1
fetch_idx = -1
for idx, arg in enumerate(list(sys.argv)):
# track index of configure and fetch_libzmq
if arg == 'configure':
configure_idx = idx
elif arg == 'fetch_libzmq':
fetch_idx = idx
if arg.startswith('--zmq='):
sys.argv.pop(idx)
if configure_idx < 0:
if fetch_idx < 0:
configure_idx = 1
else:
configure_idx = fetch_idx + 1
sys.argv.insert(configure_idx, 'configure')
sys.argv.insert(configure_idx + 1, arg)
break
for idx, arg in enumerate(list(sys.argv)):
if arg.startswith('--libzmq='):
sys.argv.remove(arg)
libzmq_name = arg.split("=", 1)[1]
if arg == '--enable-drafts':
sys.argv.remove(arg)
os.environ['ZMQ_DRAFT_API'] = '1'
if not sys.platform.startswith('win'):
cxx_flags = os.getenv("CXXFLAGS", "")
if "-std" not in cxx_flags:
cxx_flags = "-std=c++11 " + cxx_flags
os.environ["CXXFLAGS"] = cxx_flags
if cxx_flags:
# distutils doesn't support $CXXFLAGS
cxx = os.getenv("CXX", get_config_var("CXX"))
# get_config_var is broken on some old versions of pypy, add a fallback
if cxx is None:
cxx = "c++ -pthread"
os.environ["CXX"] = cxx + " " + cxx_flags
# -----------------------------------------------------------------------------
# Configuration (adapted from h5py: https://www.h5py.org/)
# -----------------------------------------------------------------------------
# --- compiler settings -------------------------------------------------
def bundled_settings(cmd):
"""settings for linking extensions against bundled libzmq"""
settings = {}
settings['libraries'] = []
settings['library_dirs'] = []
settings['include_dirs'] = [pjoin("bundled", "zeromq", "include")]
settings['runtime_library_dirs'] = []
# add pthread on freebsd
# is this necessary?
if sys.platform.startswith('freebsd'):
settings['libraries'].append('pthread')
elif sys.platform.startswith('win'):
# link against libzmq in build dir:
if sys.version_info < (3, 9, 2):
# bpo-39825: EXT_SUFFIX is wrong from sysconfig prior to 3.9.2 / 3.8.7
import distutils.sysconfig
ext_suffix = distutils.sysconfig.get_config_var("EXT_SUFFIX")
else:
ext_suffix = get_config_var("EXT_SUFFIX")
suffix = os.path.splitext(ext_suffix)[0]
print(f"DEBUG EXT_SUFFIX {suffix!r} {ext_suffix!r}")
settings['libraries'].append(libzmq_name + suffix)
settings['library_dirs'].append(pjoin(cmd.build_temp, 'buildutils'))
return settings
def check_pkgconfig():
"""pull compile / link flags from pkg-config if present."""
pcfg = None
zmq_config = None
try:
pkg_config = os.environ.get('PKG_CONFIG', 'pkg-config')
check_call([pkg_config, '--exists', 'libzmq'])
# this would arguably be better with --variable=libdir /
# --variable=includedir, but would require multiple calls
pcfg = Popen(
[pkg_config, '--libs', '--cflags', 'libzmq'], stdout=subprocess.PIPE
)
except OSError as osexception:
if osexception.errno == errno.ENOENT:
info('pkg-config not found')
else:
warn("Running pkg-config failed - %s." % osexception)
except CalledProcessError:
info("Did not find libzmq via pkg-config.")
if pcfg is not None:
output, _ = pcfg.communicate()
output = output.decode('utf8', 'replace')
bits = output.strip().split()
zmq_config = {'library_dirs': [], 'include_dirs': [], 'libraries': []}
for tok in bits:
if tok.startswith("-L"):
zmq_config['library_dirs'].append(tok[2:])
if tok.startswith("-I"):
zmq_config['include_dirs'].append(tok[2:])
if tok.startswith("-l"):
zmq_config['libraries'].append(tok[2:])
info("Settings obtained from pkg-config: %r" % zmq_config)
return zmq_config
def _add_rpath(settings, path):
"""Add rpath to settings
Implemented here because setuptools runtime_library_dirs doesn't do anything on darwin
"""
if sys.platform == 'darwin':
settings['extra_link_args'].extend(['-Wl,-rpath', '-Wl,%s' % path])
else:
settings['runtime_library_dirs'].append(path)
def settings_from_prefix(prefix=None):
"""load appropriate library/include settings from ZMQ prefix"""
settings = {}
settings['libraries'] = []
settings['include_dirs'] = []
settings['library_dirs'] = []
settings['runtime_library_dirs'] = []
settings['extra_link_args'] = []
if sys.platform.startswith('win'):
global libzmq_name
if prefix:
# add prefix itself as well, for e.g. libzmq Windows releases
for include_dir in [pjoin(prefix, 'include'), prefix]:
if os.path.exists(pjoin(include_dir, "zmq.h")):
settings['include_dirs'].append(include_dir)
info(f"Found zmq.h in {include_dir}")
break
else:
warn(f"zmq.h not found in {prefix} or {prefix}/include")
for library_dir in [pjoin(prefix, 'lib'), prefix]:
matches = glob(pjoin(library_dir, f"{libzmq_name}*.dll"))
if matches:
libzmq_path = matches[0]
libzmq_lib, libzmq_dll_name = os.path.split(libzmq_path)
libzmq_name, _ = os.path.splitext(libzmq_dll_name)
info(f"Found {libzmq_path} in {libzmq_lib}")
if libzmq_lib not in os.environ["PATH"].split(os.pathsep):
info(f"Adding {libzmq_lib} to $PATH")
os.environ["PATH"] += os.pathsep + libzmq_lib
settings['library_dirs'].append(library_dir)
break
else:
warn(f"{libzmq_name}.dll not found in {prefix} or {prefix}/lib")
settings['libraries'].append(libzmq_name)
else:
# add pthread on freebsd
if sys.platform.startswith('freebsd'):
settings['libraries'].append('pthread')
if sys.platform.startswith('sunos'):
if platform.architecture()[0] == '32bit':
settings['extra_link_args'] += ['-m32']
else:
settings['extra_link_args'] += ['-m64']
if prefix:
settings['libraries'].append('zmq')
settings['include_dirs'] += [pjoin(prefix, 'include')]
if (
sys.platform.startswith('sunos')
and platform.architecture()[0] == '64bit'
):
settings['library_dirs'] += [pjoin(prefix, 'lib/amd64')]
settings['library_dirs'] += [pjoin(prefix, 'lib')]
else:
# If prefix is not explicitly set, pull it from pkg-config by default.
# this is probably applicable across platforms, but i don't have
# sufficient test environments to confirm
pkgcfginfo = check_pkgconfig()
if pkgcfginfo is not None:
# we can get all the zmq-specific values from pkgconfg
for key, value in pkgcfginfo.items():
settings[key].extend(value)
else:
settings['libraries'].append('zmq')
if sys.platform == 'darwin' and os.path.isdir('/opt/local/lib'):
# allow macports default
settings['include_dirs'] += ['/opt/local/include']
settings['library_dirs'] += ['/opt/local/lib']
if os.environ.get('VIRTUAL_ENV', None):
# find libzmq installed in virtualenv
env = os.environ['VIRTUAL_ENV']
settings['include_dirs'] += [pjoin(env, 'include')]
settings['library_dirs'] += [pjoin(env, 'lib')]
for path in settings['library_dirs']:
_add_rpath(settings, os.path.abspath(path))
info(settings)
return settings
class LibZMQVersionError(Exception):
pass
# -----------------------------------------------------------------------------
# Extra commands
# -----------------------------------------------------------------------------
class bdist_egg_disabled(bdist_egg):
"""Disabled version of bdist_egg
Prevents setup.py install from performing setuptools' default easy_install,
which it should never ever do.
"""
def run(self):
sys.exit(
"Aborting implicit building of eggs. Use `pip install .` to install from source."
)
class Configure(build_ext):
"""Configure command adapted from h5py"""
description = "Discover ZMQ version and features"
user_options = build_ext.user_options + [
('zmq=', None, "libzmq install prefix"),
(
'build-base=',
'b',
"base directory for build library",
), # build_base from build
]
def initialize_options(self):
super().initialize_options()
self.zmq = os.environ.get("ZMQ_PREFIX") or None
self.build_base = 'build'
def finalize_options(self):
super().finalize_options()
self.tempdir = pjoin(self.build_temp, 'scratch')
self.has_run = False
self.config = discover_settings(self.build_base)
if self.zmq is not None:
merge(self.config, config_from_prefix(self.zmq))
# ensure vcredist is on PATH in MSVC toolchain
if sys.platform.startswith("win") and 'MSC' in sys.version:
locate_vcredist_dir(self.plat_name)
# need a dummy extension for run to set up a compiler
self.extensions = [Extension("fake", ["unused.c"])]
def save_config(self, name, cfg):
"""write config to JSON"""
save_config(name, cfg, self.build_base)
# write to zmq.utils.[name].json
save_config(name, cfg, os.path.join('zmq', 'utils'))
# also write to build_lib, because we might be run after copying to
# build_lib has already happened.
build_lib_utils = os.path.join(self.build_lib, 'zmq', 'utils')
if os.path.exists(build_lib_utils):
save_config(name, cfg, build_lib_utils)
def init_settings_from_config(self):
"""set up compiler settings, based on config"""
cfg = self.config
if cfg['libzmq_extension']:
settings = bundled_settings(self)
else:
settings = settings_from_prefix(cfg['zmq_prefix'])
if 'have_sys_un_h' not in cfg:
# don't link against anything when checking for sys/un.h
minus_zmq = copy.deepcopy(settings)
try:
minus_zmq['libraries'] = []
except Exception:
pass
try:
compile_and_forget(
self.tempdir,
pjoin('buildutils', 'check_sys_un.c'),
compiler=self.compiler,
**minus_zmq,
)
except Exception as e:
warn("No sys/un.h, IPC_PATH_MAX_LEN will be undefined: %s" % e)
cfg['have_sys_un_h'] = False
else:
cfg['have_sys_un_h'] = True
self.save_config('config', cfg)
settings.setdefault('define_macros', [])
if cfg['have_sys_un_h']:
settings['define_macros'].append(('HAVE_SYS_UN_H', 1))
if cfg['win_ver']:
# set target minimum Windows version
settings['define_macros'].extend(
[
('WINVER', cfg['win_ver']),
('_WIN32_WINNT', cfg['win_ver']),
]
)
if cfg.get('zmq_draft_api'):
settings['define_macros'].append(('ZMQ_BUILD_DRAFT_API', 1))
use_static_zmq = cfg.get('use_static_zmq', 'False').upper()
if use_static_zmq in ('TRUE', '1'):
settings['define_macros'].append(('ZMQ_STATIC', '1'))
if os.environ.get("PYZMQ_CYTHON_COVERAGE"):
settings['define_macros'].append(('CYTHON_TRACE', '1'))
# include internal directories
settings.setdefault('include_dirs', [])
settings['include_dirs'] += [pjoin('zmq', sub) for sub in ('utils',)]
settings.setdefault('libraries', [])
# Explicitly link dependencies, not necessary if zmq is dynamic
if sys.platform.startswith('win'):
settings['libraries'].extend(('ws2_32', 'iphlpapi', 'advapi32'))
for ext in self.distribution.ext_modules:
if ext.name.startswith('zmq.lib'):
continue
for attr, value in settings.items():
setattr(ext, attr, value)
self.compiler_settings = settings
self.save_config('compiler', settings)
def create_tempdir(self):
self.erase_tempdir()
os.makedirs(self.tempdir)
def erase_tempdir(self):
try:
shutil.rmtree(self.tempdir)
except Exception:
pass
@property
def compiler_type(self):
return self.compiler.compiler_type
@property
def cross_compiling(self):
return self.config['bdist_egg'].get('plat-name', sys.platform) != sys.platform
def check_zmq_version(self):
"""check the zmq version"""
cfg = self.config
# build test program
zmq_prefix = cfg['zmq_prefix']
detected = self.test_build(zmq_prefix, self.compiler_settings)
# now check the libzmq version
vers = tuple(detected['vers'])
vs = v_str(vers)
if cfg['allow_legacy_libzmq']:
min_zmq = min_legacy_zmq
else:
min_zmq = min_good_zmq
if vers < min_zmq:
msg = [
"Detected ZMQ version: %s, but require ZMQ >= %s"
% (vs, v_str(min_zmq)),
]
if zmq_prefix:
msg.append(" ZMQ_PREFIX=%s" % zmq_prefix)
if vers >= min_legacy_zmq:
msg.append(
" Explicitly allow legacy zmq by specifying `ZMQ_PREFIX=/zmq/prefix`"
)
raise LibZMQVersionError('\n'.join(msg))
if vers < min_good_zmq:
msg = [
"Detected legacy ZMQ version: %s. It is STRONGLY recommended to use ZMQ >= %s"
% (vs, v_str(min_good_zmq)),
]
if zmq_prefix:
msg.append(" ZMQ_PREFIX=%s" % zmq_prefix)
warn('\n'.join(msg))
elif vers < target_zmq:
warn(
"Detected ZMQ version: %s, but pyzmq targets ZMQ %s."
% (vs, v_str(target_zmq))
)
warn(
"libzmq features and fixes introduced after %s will be unavailable."
% vs
)
line()
elif vers >= dev_zmq:
warn(
"Detected ZMQ version: %s. Some new features in libzmq may not be exposed by pyzmq."
% vs
)
line()
if sys.platform.startswith('win'):
# fetch libzmq.dll into local dir
local_dll = localpath('zmq', libzmq_name + '.dll')
if not zmq_prefix and not os.path.exists(local_dll):
fatal(
"ZMQ directory must be specified on Windows via setup.cfg or 'ZMQ_PREFIX=/path/to/libzmq' env"
)
def bundle_libzmq_extension(self):
bundledir = "bundled"
ext_modules = self.distribution.ext_modules
if ext_modules and any(m.name == 'zmq.libzmq' for m in ext_modules):
# I've already been run
return
line()
info("Using bundled libzmq")
# fetch sources for libzmq extension:
if not os.path.exists(bundledir):
os.makedirs(bundledir)
fetch_libzmq(bundledir)
stage_platform_hpp(pjoin(bundledir, 'zeromq'))
sources = [pjoin('buildutils', 'initlibzmq.cpp')]
sources.extend(
[
src
for src in glob(pjoin(bundledir, 'zeromq', 'src', '*.cpp'))
# exclude draft ws transport files
if not os.path.basename(src).startswith(("ws_", "wss_"))
]
)
if sys.platform.startswith("win"): # only compile wepoll on windows...
sources.append(pjoin('bundled', 'zeromq', 'external', 'wepoll', 'wepoll.c'))
includes = [pjoin(bundledir, 'zeromq', 'include')]
if bundled_version < (4, 2, 0):
tweetnacl = pjoin(bundledir, 'zeromq', 'tweetnacl')
tweetnacl_sources = glob(pjoin(tweetnacl, 'src', '*.c'))
randombytes = pjoin(tweetnacl, 'contrib', 'randombytes')
if sys.platform.startswith('win'):
tweetnacl_sources.append(pjoin(randombytes, 'winrandom.c'))
else:
tweetnacl_sources.append(pjoin(randombytes, 'devurandom.c'))
sources += tweetnacl_sources
includes.append(pjoin(tweetnacl, 'src'))
includes.append(randombytes)
else:
# >= 4.2
sources += glob(pjoin(bundledir, 'zeromq', 'src', 'tweetnacl.c'))
# construct the Extensions:
libzmq = Extension(
'zmq.libzmq',
sources=sources,
include_dirs=includes,
)
# register the extension:
# doing this here means we must be run
# before finalize_options in build_ext
self.distribution.ext_modules.insert(0, libzmq)
# use tweetnacl to provide CURVE support
libzmq.define_macros.append(('ZMQ_HAVE_CURVE', 1))
libzmq.define_macros.append(('ZMQ_USE_TWEETNACL', 1))
# set draft flag
if self.config["zmq_draft_api"]:
libzmq.define_macros.append(('ZMQ_BUILD_DRAFT_API', 1))
# select polling subsystem based on platform
if sys.platform == "darwin" or "bsd" in sys.platform:
libzmq.define_macros.append(('ZMQ_USE_KQUEUE', 1))
libzmq.define_macros.append(('ZMQ_IOTHREADS_USE_KQUEUE', 1))
libzmq.define_macros.append(('ZMQ_POLL_BASED_ON_POLL', 1))
elif 'linux' in sys.platform:
libzmq.define_macros.append(('ZMQ_USE_EPOLL', 1))
libzmq.define_macros.append(('ZMQ_IOTHREADS_USE_EPOLL', 1))
libzmq.define_macros.append(('ZMQ_POLL_BASED_ON_POLL', 1))
elif sys.platform.startswith('win'):
libzmq.define_macros.append(('ZMQ_USE_SELECT', 1))
libzmq.define_macros.append(('ZMQ_IOTHREADS_USE_SELECT', 1))
libzmq.define_macros.append(('ZMQ_POLL_BASED_ON_SELECT', 1))
else:
# this may not be sufficiently precise
libzmq.define_macros.append(('ZMQ_USE_POLL', 1))
libzmq.define_macros.append(('ZMQ_IOTHREADS_USE_POLL', 1))
libzmq.define_macros.append(('ZMQ_POLL_BASED_ON_POLL', 1))
if sys.platform.startswith('win'):
# include defines from zeromq msvc project:
libzmq.define_macros.append(('FD_SETSIZE', 16384))
libzmq.define_macros.append(('DLL_EXPORT', 1))
libzmq.define_macros.append(('_CRT_SECURE_NO_WARNINGS', 1))
# When compiling the C++ code inside of libzmq itself, we want to
# avoid "warning C4530: C++ exception handler used, but unwind
# semantics are not enabled. Specify /EHsc".
if self.compiler.compiler_type == 'msvc':
libzmq.extra_compile_args.append('/EHsc')
elif self.compiler.compiler_type == 'mingw32':
libzmq.define_macros.append(('ZMQ_HAVE_MINGW32', 1))
# And things like sockets come from libraries that must be named.
libzmq.libraries.extend(['rpcrt4', 'ws2_32', 'advapi32', 'iphlpapi'])
else:
libzmq.include_dirs.append(bundledir)
# check if we need to link against Realtime Extensions library
if not sys.platform.startswith(('darwin', 'freebsd')):
line()
info("checking for timer_create")
if not self.compiler.has_function('timer_create'):
info("no timer_create, linking librt")
libzmq.libraries.append('rt')
else:
info("ok")
# copy the header files to the source tree.
bundledincludedir = pjoin('zmq', 'include')
if not os.path.exists(bundledincludedir):
os.makedirs(bundledincludedir)
if not os.path.exists(pjoin(self.build_lib, bundledincludedir)):
os.makedirs(pjoin(self.build_lib, bundledincludedir))
for header in glob(pjoin(bundledir, 'zeromq', 'include', '*.h')):
shutil.copyfile(header, pjoin(bundledincludedir, basename(header)))
shutil.copyfile(
header, pjoin(self.build_lib, bundledincludedir, basename(header))
)
# update other extensions, with bundled settings
self.config['libzmq_extension'] = True
self.init_settings_from_config()
self.save_config('config', self.config)
def fallback_on_bundled(self):
"""Couldn't build, fallback after waiting a while"""
line()
warn(
'\n'.join(
[
"Couldn't find an acceptable libzmq on the system.",
"",
"If you expected pyzmq to link against an installed libzmq, please check to make sure:",
"",
" * You have a C compiler installed",
" * A development version of Python is installed (including headers)",
" * A development version of ZMQ >= %s is installed (including headers)"
% v_str(min_good_zmq),
" * If ZMQ is not in a default location, specify the env ZMQ_PREFIX=<path>",
" * If you did recently install ZMQ to a default location,",
" try rebuilding the ld cache with `sudo ldconfig`",
" or specify zmq's location with `ZMQ_PREFIX=/usr/local`",
"",
]
)
)
info(
'\n'.join(
[
"You can skip all this detection/waiting nonsense if you know",
"you want pyzmq to bundle libzmq as an extension by passing:",
"",
" `ZMQ_PREFIX=bundled`",
"",
"I will now try to build libzmq as a Python extension",
"unless you interrupt me (^C) in the next 10 seconds...",
"",
]
)
)
for i in range(10, 0, -1):
sys.stdout.write('\r%2i...' % i)
sys.stdout.flush()
time.sleep(1)
info("")
return self.bundle_libzmq_extension()
def test_build(self, prefix, settings):
"""do a test build ob libzmq"""
self.create_tempdir()
settings = settings.copy()
line()
info("Configure: Autodetecting ZMQ settings...")
info(" Custom ZMQ dir: %s" % prefix)
try:
detected = detect_zmq(self.tempdir, compiler=self.compiler, **settings)
finally:
self.erase_tempdir()
info(" ZMQ version detected: %s" % v_str(detected['vers']))
return detected
def finish_run(self):
self.save_config('config', self.config)
line()
def build_extensions(self):
"""Need an empty build_extensions so that .run() gives us a configured compiler"""
def run(self):
# super().run() is what sets up self.compiler
super().run()
self.init_settings_from_config()
cfg = self.config
if cfg['libzmq_extension']:
self.bundle_libzmq_extension()
self.finish_run()
return
# When cross-compiling and zmq is given explicitly, we can't testbuild
# (as we can't testrun the binary), we assume things are alright.
if cfg['skip_check_zmq'] or self.cross_compiling:
warn("Skipping zmq version check")
self.finish_run()
return
zmq_prefix = cfg['zmq_prefix']
# There is no available default on Windows, so start with fallback unless
# zmq was given explicitly, or libzmq extension was explicitly prohibited.
if (
sys.platform.startswith("win")
and not cfg['no_libzmq_extension']
and not zmq_prefix
):
self.fallback_on_bundled()
self.finish_run()
return
# first try with given config or defaults
try:
self.check_zmq_version()
except LibZMQVersionError as e:
info("\nBad libzmq version: %s\n" % e)
except Exception as e:
# print the error as setuptools would if we let it raise:
info("\nerror: %s\n" % e)
else:
self.finish_run()
return
# try fallback on /usr/local on *ix if no prefix is given
if not zmq_prefix and not sys.platform.startswith('win'):
info("Failed with default libzmq, trying again with /usr/local")
time.sleep(1)
zmq_prefix = cfg['zmq_prefix'] = '/usr/local'
self.init_settings_from_config()
try:
self.check_zmq_version()
except LibZMQVersionError as e:
info("\nBad libzmq version: %s\n" % e)
except Exception as e:
# print the error as setuptools would if we let it raise:
info("\nerror: %s\n" % e)
else:
# if we get here the second run succeeded, so we need to update compiler
# settings for the extensions with /usr/local prefix
self.finish_run()
return
# finally, fallback on bundled
if cfg['no_libzmq_extension']:
fatal(
"Falling back on bundled libzmq,"
" but config has explicitly prohibited building the libzmq extension."
)
self.fallback_on_bundled()
self.finish_run()
class FetchCommand(Command):
"""Fetch libzmq, that's it."""
description = "Fetch libzmq sources or dll"
user_options = [
('dll', None, "Fetch binary dll (Windows only)"),
]
def initialize_options(self):
self.dll = False
def finalize_options(self):
pass
def run(self):
# fetch sources for libzmq extension:
if self.dll:
self.fetch_libzmq_dll()
else:
self.fetch_libzmq_src()
def fetch_libzmq_dll(self):
libdir = "libzmq-dll"
if os.path.exists(libdir):
info("Scrubbing directory: %s" % libdir)
shutil.rmtree(libdir)
if not os.path.exists(libdir):
os.makedirs(libdir)
fetch_libzmq_dll(libdir)
for archive in glob(pjoin(libdir, '*.zip')):
os.remove(archive)
def fetch_libzmq_src(self):
bundledir = "bundled"
if os.path.exists(bundledir):
info("Scrubbing directory: %s" % bundledir)
shutil.rmtree(bundledir)
if not os.path.exists(bundledir):
os.makedirs(bundledir)
fetch_libzmq(bundledir)
class TestCommand(Command):
"""Custom setuptools command to run the test suite."""
description = "DEPRECATED, use pytest"
user_options = []
def initialize_options(self):
self._dir = os.getcwd()
def finalize_options(self):
pass
def run(self):
"""Run the test suite with pytest"""
warn("Running pyzmq's tests with `setup.py test` is deprecated. Use `pytest`.")
time.sleep(10)
# crude check for inplace build:
try:
import zmq
except ImportError:
print_exc()
fatal(
'\n '.join(
[
"Could not import zmq!",
"You must build pyzmq with 'python setup.py build_ext --inplace' for 'python setup.py test' to work.",
"If you did build pyzmq in-place, then this is a real error.",
]
)
)
sys.exit(1)
info(f"Testing pyzmq-{zmq.pyzmq_version()} with libzmq-{zmq.zmq_version()}")
p = Popen([sys.executable, '-m', 'pytest', '-v', os.path.join('zmq', 'tests')])
p.wait()
sys.exit(p.returncode)
class GitRevisionCommand(Command):
"""find the current git revision and add it to zmq.sugar.version.__revision__"""
description = "Store current git revision in version.py"
user_options = []
def initialize_options(self):
self.version_py = pjoin('zmq', 'sugar', 'version.py')
def run(self):
try:
p = Popen('git log -1'.split(), stdin=PIPE, stdout=PIPE, stderr=PIPE)
except OSError:
warn("No git found, skipping git revision")
return
if p.wait():
warn("checking git branch failed")
info(p.stderr.read())
return
line = p.stdout.readline().decode().strip()
if not line.startswith('commit'):
warn("bad commit line: %r" % line)
return
rev = line.split()[-1]
# now that we have the git revision, we can apply it to version.py
with open(self.version_py) as f:
lines = f.readlines()
for i, line in enumerate(lines):
if line.startswith('__revision__'):
lines[i] = "__revision__ = '%s'\n" % rev
break
with open(self.version_py, 'w') as f:
f.writelines(lines)
def finalize_options(self):
pass
class CleanCommand(Command):
"""Custom setuptools command to clean the .so and .pyc files."""
user_options = [
('all', 'a', "remove all build output, not just temporary by-products")
]
boolean_options = ['all']
def initialize_options(self):
self.all = None
def finalize_options(self):
pass
def run(self):
_clean_me = []
_clean_trees = []
for d in ('build', 'dist', 'conf'):
if os.path.exists(d):
_clean_trees.append(d)
for root, dirs, files in os.walk('buildutils'):
if any(root.startswith(pre) for pre in _clean_trees):
continue
for f in files:
if os.path.splitext(f)[-1] == '.pyc':
_clean_me.append(pjoin(root, f))
if '__pycache__' in dirs:
_clean_trees.append(pjoin(root, '__pycache__'))
for root, dirs, files in os.walk('zmq'):
if any(root.startswith(pre) for pre in _clean_trees):
continue
for f in files:
if os.path.splitext(f)[-1] in ('.pyc', '.so', '.o', '.pyd', '.json'):
_clean_me.append(pjoin(root, f))
# remove generated cython files
if self.all:
for f in files:
f2 = os.path.splitext(f)
if f2[1] == '.c' and os.path.isfile(
os.path.join(root, f2[0]) + '.pyx'
):
_clean_me.append(pjoin(root, f))
for d in dirs:
if d == '__pycache__':
_clean_trees.append(pjoin(root, d))
bundled = glob(pjoin('zmq', 'libzmq*'))
_clean_me.extend([b for b in bundled if b not in _clean_me])
bundled_headers = glob(pjoin('zmq', 'include', '*.h'))
_clean_me.extend([h for h in bundled_headers if h not in _clean_me])
for clean_me in _clean_me:
print("removing %s" % clean_me)
try:
os.unlink(clean_me)
except Exception as e:
print(e, file=sys.stderr)
for clean_tree in _clean_trees:
print("removing %s/" % clean_tree)
try:
shutil.rmtree(clean_tree)
except Exception as e:
print(e, file=sys.stderr)
class CheckSDist(sdist):
"""Custom sdist that ensures Cython has compiled all pyx files to c."""
def initialize_options(self):
sdist.initialize_options(self)
self._pyxfiles = []
for root, dirs, files in os.walk('zmq'):
for f in files:
if f.endswith('.pyx'):
self._pyxfiles.append(pjoin(root, f))
def run(self):
self.run_command('fetch_libzmq')
if 'cython' in cmdclass:
self.run_command('cython')
else:
for pyxfile in self._pyxfiles:
cfile = pyxfile[:-3] + 'c'
msg = (
"C-source file '%s' not found." % (cfile)
+ " Run 'setup.py cython' before sdist."
)
assert os.path.isfile(cfile), msg
sdist.run(self)
@contextmanager
def use_cxx(compiler):
"""use C++ compiler in this context
used in fix_cxx which detects when C++ should be used
"""
compiler_so_save = compiler.compiler_so[:]
compiler_so_cxx = compiler.compiler_cxx + compiler.compiler_so[1:]
# actually use CXX compiler
compiler.compiler_so = compiler_so_cxx
try:
yield
finally:
# restore original state
compiler.compiler_so = compiler_so_save
@contextmanager
def fix_cxx(compiler, extension):
"""Fix C++ compilation to use C++ compiler
See https://bugs.python.org/issue1222585 for Python support for C++,
which apparently doesn't exist and only works by accident.
"""
if compiler.detect_language(extension.sources) != "c++":
# no c++, nothing to do
yield
return
_compile_save = compiler._compile
def _compile_cxx(obj, src, ext, *args, **kwargs):
if compiler.language_map.get(ext) == "c++":
with use_cxx(compiler):
_compile_save(obj, src, ext, *args, **kwargs)
else:
_compile_save(obj, src, ext, *args, **kwargs)
compiler._compile = _compile_cxx
try:
yield
finally:
compiler._compile = _compile_save
class CheckingBuildExt(build_ext):
"""Subclass build_ext to get clearer report if Cython is necessary."""
def check_cython_extensions(self, extensions):
for ext in extensions:
for src in ext.sources:
if not os.path.exists(src):
fatal(
"""Cython-generated file '%s' not found.
Cython >= %s is required to compile pyzmq from a development branch.
Please install Cython or download a release package of pyzmq.
"""
% (src, min_cython_version)
)
def build_extensions(self):
self.check_cython_extensions(self.extensions)
self.check_extensions_list(self.extensions)
if self.compiler.compiler_type == 'mingw32':
customize_mingw(self.compiler)
for ext in self.extensions:
self.build_extension(ext)
def build_extension(self, ext):
with fix_cxx(self.compiler, ext):
super().build_extension(ext)
ext_path = self.get_ext_fullpath(ext.name)
patch_lib_paths(ext_path, self.compiler.library_dirs)
def finalize_options(self):
# check version, to prevent confusing undefined constant errors
self.distribution.run_command("configure")
return super().finalize_options()
class ConstantsCommand(Command):
"""Rebuild templated files for constants
To be run after adding new constants to `utils/constant_names`.
"""
user_options = []
def initialize_options(self):
return
def finalize_options(self):
pass
def run(self):
from buildutils.constants import render_constants
render_constants()
cmdclass = {
"bdist_egg": bdist_egg if "bdist_egg" in sys.argv else bdist_egg_disabled,
"clean": CleanCommand,
"configure": Configure,
"constants": ConstantsCommand,
"fetch_libzmq": FetchCommand,
"revision": GitRevisionCommand,
"sdist": CheckSDist,
"test": TestCommand,
}
# -----------------------------------------------------------------------------
# Extensions
# -----------------------------------------------------------------------------
def makename(path, ext):
return os.path.abspath(pjoin('zmq', *path)) + ext
pxd = lambda *path: makename(path, '.pxd')
pxi = lambda *path: makename(path, '.pxi')
pyx = lambda *path: makename(path, '.pyx')
dotc = lambda *path: makename(path, '.c')
doth = lambda *path: makename(path, '.h')
libzmq = pxd('backend', 'cython', 'libzmq')
buffers = pxd('utils', 'buffers')
message = pxd('backend', 'cython', 'message')
context = pxd('backend', 'cython', 'context')
socket = pxd('backend', 'cython', 'socket')
checkrc = pxd('backend', 'cython', 'checkrc')
monqueue = pxd('devices', 'monitoredqueue')
mutex = doth('utils', 'mutex')
submodules = {
'backend.cython': {
'error': [libzmq, checkrc],
'_poll': [libzmq, socket, context, checkrc],
'utils': [libzmq, checkrc],
'context': [context, libzmq, checkrc],
'message': [libzmq, buffers, message, checkrc, mutex],
'socket': [context, message, socket, libzmq, buffers, checkrc],
'_device': [libzmq, socket, context, checkrc],
'_proxy_steerable': [libzmq, socket, checkrc],
'_version': [libzmq],
},
'devices': {
'monitoredqueue': [buffers, libzmq, monqueue, socket, context, checkrc],
},
}
# require cython 0.29
min_cython_version = "0.29"
cython_language_level = "3str"
try:
import Cython
if V(Cython.__version__) < V(min_cython_version):
raise ImportError(
"Cython >= %s required for cython build, found %s"
% (min_cython_version, Cython.__version__)
)
from Cython.Build import cythonize
from Cython.Distutils.build_ext import new_build_ext as build_ext_cython
cython = True
except Exception:
cython = False
suffix = '.c'
cmdclass['build_ext'] = CheckingBuildExt
class MissingCython(Command):
user_options = []
def initialize_options(self):
pass
def finalize_options(self):
pass
def run(self):
try:
import Cython
except ImportError:
warn("Cython is missing")
else:
cv = getattr(Cython, "__version__", None)
if cv is None or V(cv) < V(min_cython_version):
warn(
"Cython >= %s is required for compiling Cython sources, "
"found: %s" % (min_cython_version, cv or Cython)
)
cmdclass['cython'] = MissingCython
else:
suffix = '.pyx'
class CythonCommand(build_ext_cython):
"""Custom setuptools command subclassed from Cython.Distutils.build_ext
to compile pyx->c, and stop there. All this does is override the
C-compile method build_extension() with a no-op."""
description = "Compile Cython sources to C"
def build_extension(self, ext):
pass
class zbuild_ext(build_ext_cython):
def build_extensions(self):
if self.compiler.compiler_type == 'mingw32':
customize_mingw(self.compiler)
return super().build_extensions()
def build_extension(self, ext):
with fix_cxx(self.compiler, ext):
super().build_extension(ext)
ext_path = self.get_ext_fullpath(ext.name)
patch_lib_paths(ext_path, self.compiler.library_dirs)
def finalize_options(self):
self.distribution.run_command("configure")
return super().finalize_options()
cmdclass["cython"] = CythonCommand
cmdclass["build_ext"] = zbuild_ext
extensions = []
ext_include_dirs = [pjoin('zmq', sub) for sub in ('utils',)]
ext_kwargs = {
'include_dirs': ext_include_dirs,
}
for submod, packages in submodules.items():
for pkg in sorted(packages):
sources = [pjoin("zmq", submod.replace(".", os.path.sep), pkg + suffix)]
ext = Extension(f"zmq.{submod}.{pkg}", sources=sources, **ext_kwargs)
extensions.append(ext)
if cython:
# set binding so that compiled methods can be inspected
# set language-level to 3str, requires Cython 0.29
cython_directives = {"binding": True, "language_level": "3str"}
if os.environ.get("PYZMQ_CYTHON_COVERAGE"):
cython_directives["linetrace"] = True
extensions = cythonize(extensions, compiler_directives=cython_directives)
if pypy:
extensions = []
if pypy or os.environ.get("PYZMQ_BACKEND_CFFI"):
cffi_modules = ['buildutils/build_cffi.py:ffi']
else:
cffi_modules = []
package_data = {
'zmq': ['*.pxd', '*.pyi', '*' + lib_ext, 'py.typed'],
'zmq.backend': ['*.pyi'],
'zmq.backend.cython': ['*.pxd', '*.pxi'],
'zmq.backend.cffi': ['*.h', '*.c'],
'zmq.devices': ['*.pxd'],
'zmq.sugar': ['*.pyi'],
'zmq.tests': ['*.pyx'],
'zmq.utils': ['*.pxd', '*.h', '*.json'],
}
def find_packages():
"""adapted from IPython's setupbase.find_packages()"""
packages = []
for dir, subdirs, files in os.walk('zmq'):
package = dir.replace(os.path.sep, '.')
if '__init__.py' not in files:
# not a package
continue
packages.append(package)
return packages
# -----------------------------------------------------------------------------
# Main setup
# -----------------------------------------------------------------------------
with open('README.md', encoding='utf-8') as f:
long_desc = f.read()
setup_args = dict(
name="pyzmq",
version="25.0.0",
packages=find_packages(),
ext_modules=extensions,
cffi_modules=cffi_modules,
package_data=package_data,
author="Brian E. Granger, Min Ragan-Kelley",
author_email="zeromq-dev@lists.zeromq.org",
url="https://pyzmq.readthedocs.org",
project_urls={
'Source': 'https://github.com/zeromq/pyzmq',
},
description="Python bindings for 0MQ",
long_description=long_desc,
long_description_content_type="text/markdown",
license="LGPL+BSD",
cmdclass=cmdclass,
classifiers=[
"Development Status :: 5 - Production/Stable",
"Intended Audience :: Developers",
"Intended Audience :: Science/Research",
"Intended Audience :: System Administrators",
"License :: OSI Approved :: GNU Library or Lesser General Public License (LGPL)",
"License :: OSI Approved :: BSD License",
"Operating System :: MacOS :: MacOS X",
"Operating System :: Microsoft :: Windows",
"Operating System :: POSIX",
"Topic :: System :: Networking",
"Programming Language :: Python :: 3",
"Programming Language :: Python :: 3 :: Only",
"Programming Language :: Python :: 3.6",
"Programming Language :: Python :: 3.7",
"Programming Language :: Python :: 3.8",
"Programming Language :: Python :: 3.9",
],
zip_safe=False,
python_requires=">=3.6",
install_requires=[
"cffi; implementation_name == 'pypy'",
],
)
if not os.path.exists(os.path.join("zmq", "backend", "cython", "socket.c")):
# this generally means pip install from git
# which requires Cython
setup_args.setdefault("setup_requires", []).append(
f"cython>={min_cython_version}; implementation_name == 'cpython'",
)
setup(**setup_args)