blob: cdd3bb3f0dd4d5763b2b67d337ad42c5a975e9c2 [file] [log] [blame]
Craig Tiller6169d5f2016-03-31 07:46:18 -07001# Copyright 2015, Google Inc.
Nathaniel Manistab68f3d12015-01-23 21:32:47 +00002# All rights reserved.
3#
4# Redistribution and use in source and binary forms, with or without
5# modification, are permitted provided that the following conditions are
6# met:
7#
8# * Redistributions of source code must retain the above copyright
9# notice, this list of conditions and the following disclaimer.
10# * Redistributions in binary form must reproduce the above
11# copyright notice, this list of conditions and the following disclaimer
12# in the documentation and/or other materials provided with the
13# distribution.
14# * Neither the name of Google Inc. nor the names of its
15# contributors may be used to endorse or promote products derived from
16# this software without specific prior written permission.
17#
18# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
19# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
20# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
21# A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
22# OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
23# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
24# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
25# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
26# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
27# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
28# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29
Nathaniel Manistab8b0adf2015-01-29 00:30:51 +000030"""A setup module for the GRPC Python package."""
Ken Payson5998cd72016-08-10 15:39:43 -070031from distutils import cygwinccompiler
Masood Malekghassemiabdff3d2016-07-18 13:26:25 -070032from distutils import extension as _extension
Masood Malekghassemi398b06e2016-07-15 23:17:41 -070033from distutils import util
Masood Malekghassemi743c10c2015-06-16 18:05:27 -070034import os
Masood Malekghassemid65632a2015-07-27 14:30:09 -070035import os.path
Masood Malekghassemiabdff3d2016-07-18 13:26:25 -070036import pkg_resources
Masood Malekghassemi586e3832016-06-03 19:29:12 -070037import platform
Masood Malekghassemi398b06e2016-07-15 23:17:41 -070038import re
Ken Paysonfe754b42016-06-23 12:32:07 -070039import shlex
Masood Malekghassemi6d2ef172016-01-04 15:33:17 -080040import shutil
Alexander Staubo8eac91e2015-04-06 12:06:22 -040041import sys
Ken Payson1efb6012016-06-08 13:06:44 -070042import sysconfig
Nathaniel Manistab68f3d12015-01-23 21:32:47 +000043
Masood Malekghassemi743c10c2015-06-16 18:05:27 -070044import setuptools
Masood Malekghassemib7e7b3f2015-12-17 00:16:49 -080045from setuptools.command import egg_info
46
47# Redirect the manifest template from MANIFEST.in to PYTHON-MANIFEST.in.
48egg_info.manifest_maker.template = 'PYTHON-MANIFEST.in'
Masood Malekghassemi743c10c2015-06-16 18:05:27 -070049
Leifur Halldor Asgeirsson38a3d7a2016-03-01 10:00:08 -050050PY3 = sys.version_info.major == 3
Masood Malekghassemi58a04942016-07-16 01:42:52 -070051PYTHON_STEM = os.path.join('src', 'python', 'grpcio')
52CORE_INCLUDE = ('include', '.',)
53BORINGSSL_INCLUDE = (os.path.join('third_party', 'boringssl', 'include'),)
54ZLIB_INCLUDE = (os.path.join('third_party', 'zlib'),)
Masood Malekghassemi116982e2015-12-11 15:53:38 -080055
Masood Malekghassemid65632a2015-07-27 14:30:09 -070056# Ensure we're in the proper directory whether or not we're being used by pip.
57os.chdir(os.path.dirname(os.path.abspath(__file__)))
Masood Malekghassemi58a1dc22016-01-21 14:23:55 -080058sys.path.insert(0, os.path.abspath(PYTHON_STEM))
Masood Malekghassemid65632a2015-07-27 14:30:09 -070059
Masood Malekghassemi116982e2015-12-11 15:53:38 -080060# Break import-style to ensure we can actually find our in-repo dependencies.
Ken Payson5998cd72016-08-10 15:39:43 -070061import _spawn_patch
Masood Malekghassemid65632a2015-07-27 14:30:09 -070062import commands
Masood Malekghassemi116982e2015-12-11 15:53:38 -080063import grpc_core_dependencies
Craig Tillerf008f062016-02-08 12:48:03 -080064import grpc_version
Masood Malekghassemi743c10c2015-06-16 18:05:27 -070065
Ken Payson5998cd72016-08-10 15:39:43 -070066_spawn_patch.monkeypatch_spawn()
Masood Malekghassemi586e3832016-06-03 19:29:12 -070067
Masood Malekghassemidf97aec2016-01-11 16:19:14 -080068LICENSE = '3-clause BSD'
69
Masood Malekghassemi7566c9a2015-10-21 20:29:23 -070070# Environment variable to determine whether or not the Cython extension should
71# *use* Cython or use the generated C files. Note that this requires the C files
Masood Malekghassemif17f0f62016-07-08 12:07:49 -070072# to have been generated by building first *with* Cython support. Even if this
73# is set to false, if the script detects that the generated `.c` file isn't
74# present, then it will still attempt to use Cython.
Masood Malekghassemi7566c9a2015-10-21 20:29:23 -070075BUILD_WITH_CYTHON = os.environ.get('GRPC_PYTHON_BUILD_WITH_CYTHON', False)
Masood Malekghassemi743c10c2015-06-16 18:05:27 -070076
Masood Malekghassemi7566c9a2015-10-21 20:29:23 -070077# Environment variable to determine whether or not to enable coverage analysis
78# in Cython modules.
79ENABLE_CYTHON_TRACING = os.environ.get(
80 'GRPC_PYTHON_ENABLE_CYTHON_TRACING', False)
81
Masood Malekghassemi2da46662016-10-25 15:57:48 -070082# Environment variable specifying whether or not there's interest in setting up
83# documentation building.
84ENABLE_DOCUMENTATION_BUILD = os.environ.get(
85 'GRPC_PYTHON_ENABLE_DOCUMENTATION_BUILD', False)
86
Masood Malekghassemi586e3832016-06-03 19:29:12 -070087# There are some situations (like on Windows) where CC, CFLAGS, and LDFLAGS are
88# entirely ignored/dropped/forgotten by distutils and its Cygwin/MinGW support.
89# We use these environment variables to thus get around that without locking
90# ourselves in w.r.t. the multitude of operating systems this ought to build on.
Masood Malekghassemid66be0b2016-07-07 11:24:39 -070091# We can also use these variables as a way to inject environment-specific
92# compiler/linker flags. We assume GCC-like compilers and/or MinGW as a
93# reasonable default.
94EXTRA_ENV_COMPILE_ARGS = os.environ.get('GRPC_PYTHON_CFLAGS', None)
95EXTRA_ENV_LINK_ARGS = os.environ.get('GRPC_PYTHON_LDFLAGS', None)
96if EXTRA_ENV_COMPILE_ARGS is None:
Ken Payson5998cd72016-08-10 15:39:43 -070097 EXTRA_ENV_COMPILE_ARGS = ''
98 if 'win32' in sys.platform and sys.version_info < (3, 5):
Masood Malekghassemid66be0b2016-07-07 11:24:39 -070099 # We use define flags here and don't directly add to DEFINE_MACROS below to
100 # ensure that the expert user/builder has a way of turning it off (via the
101 # envvars) without adding yet more GRPC-specific envvars.
102 # See https://sourceforge.net/p/mingw-w64/bugs/363/
103 if '32' in platform.architecture()[0]:
104 EXTRA_ENV_COMPILE_ARGS += ' -D_ftime=_ftime32 -D_timeb=__timeb32 -D_ftime_s=_ftime32_s'
105 else:
106 EXTRA_ENV_COMPILE_ARGS += ' -D_ftime=_ftime64 -D_timeb=__timeb64'
107 elif "linux" in sys.platform or "darwin" in sys.platform:
Nicolas "Pixel" Noblee3e17d32016-08-20 01:45:32 +0200108 EXTRA_ENV_COMPILE_ARGS += ' -fvisibility=hidden -fno-wrapv'
Masood Malekghassemid66be0b2016-07-07 11:24:39 -0700109if EXTRA_ENV_LINK_ARGS is None:
Ken Payson5998cd72016-08-10 15:39:43 -0700110 EXTRA_ENV_LINK_ARGS = ''
111 if "linux" in sys.platform or "darwin" in sys.platform:
112 EXTRA_ENV_LINK_ARGS += ' -lpthread'
113 elif "win32" in sys.platform and sys.version_info < (3, 5):
114 msvcr = cygwinccompiler.get_msvcr()[0]
Masood Malekghassemid66be0b2016-07-07 11:24:39 -0700115 # TODO(atash) sift through the GCC specs to see if libstdc++ can have any
116 # influence on the linkage outcome on MinGW for non-C++ programs.
117 EXTRA_ENV_LINK_ARGS += (
118 ' -static-libgcc -static-libstdc++ -mcrtdll={msvcr} '
119 '-static'.format(msvcr=msvcr))
Nicolas "Pixel" Noblee3e17d32016-08-20 01:45:32 +0200120 if "linux" in sys.platform:
Masood Malekghassemid66be0b2016-07-07 11:24:39 -0700121 EXTRA_ENV_LINK_ARGS += ' -Wl,-wrap,memcpy'
Ken Payson5998cd72016-08-10 15:39:43 -0700122
Masood Malekghassemid66be0b2016-07-07 11:24:39 -0700123EXTRA_COMPILE_ARGS = shlex.split(EXTRA_ENV_COMPILE_ARGS)
124EXTRA_LINK_ARGS = shlex.split(EXTRA_ENV_LINK_ARGS)
Masood Malekghassemi586e3832016-06-03 19:29:12 -0700125
Masood Malekghassemi7566c9a2015-10-21 20:29:23 -0700126CYTHON_EXTENSION_PACKAGE_NAMES = ()
Masood Malekghassemi5a65bcd2015-09-25 11:27:10 -0700127
Masood Malekghassemi116982e2015-12-11 15:53:38 -0800128CYTHON_EXTENSION_MODULE_NAMES = ('grpc._cython.cygrpc',)
Masood Malekghassemi5a65bcd2015-09-25 11:27:10 -0700129
Masood Malekghassemi3acddb22016-07-13 18:38:33 -0700130CYTHON_HELPER_C_FILES = ()
Masood Malekghassemidf1e07e2016-02-02 14:17:14 -0800131
Masood Malekghassemi586e3832016-06-03 19:29:12 -0700132CORE_C_FILES = tuple(grpc_core_dependencies.CORE_SOURCE_FILES)
Masood Malekghassemidf1e07e2016-02-02 14:17:14 -0800133
Masood Malekghassemi387e1162016-01-05 10:16:12 -0800134EXTENSION_INCLUDE_DIRECTORIES = (
Masood Malekghassemi0cc27922016-01-22 16:32:41 -0800135 (PYTHON_STEM,) + CORE_INCLUDE + BORINGSSL_INCLUDE + ZLIB_INCLUDE)
Nathaniel Manistaa4ead5d2015-01-31 01:33:27 +0000136
Masood Malekghassemi10509a22016-02-03 13:32:22 -0800137EXTENSION_LIBRARIES = ()
Masood Malekghassemic9c53ee2016-02-03 10:30:23 -0800138if "linux" in sys.platform:
Craig Tiller4bef7ce2016-02-02 08:38:43 -0800139 EXTENSION_LIBRARIES += ('rt',)
Masood Malekghassemi10509a22016-02-03 13:32:22 -0800140if not "win32" in sys.platform:
141 EXTENSION_LIBRARIES += ('m',)
Masood Malekghassemi586e3832016-06-03 19:29:12 -0700142if "win32" in sys.platform:
Ken Payson5998cd72016-08-10 15:39:43 -0700143 EXTENSION_LIBRARIES += ('advapi32', 'ws2_32',)
Nathaniel Manistaa4ead5d2015-01-31 01:33:27 +0000144
Masood Malekghassemi58a04942016-07-16 01:42:52 -0700145DEFINE_MACROS = (
146 ('OPENSSL_NO_ASM', 1), ('_WIN32_WINNT', 0x600),
147 ('GPR_BACKWARDS_COMPATIBILITY_MODE', 1),)
Masood Malekghassemi586e3832016-06-03 19:29:12 -0700148if "win32" in sys.platform:
Ken Payson5998cd72016-08-10 15:39:43 -0700149 DEFINE_MACROS += (('WIN32_LEAN_AND_MEAN', 1),)
Masood Malekghassemi586e3832016-06-03 19:29:12 -0700150 if '64bit' in platform.architecture()[0]:
151 DEFINE_MACROS += (('MS_WIN64', 1),)
Ken Payson5998cd72016-08-10 15:39:43 -0700152 elif sys.version_info >= (3, 5):
153 # For some reason, this is needed to get access to inet_pton/inet_ntop
154 # on msvc, but only for 32 bits
155 DEFINE_MACROS += (('NTDDI_VERSION', 0x06000000),)
Masood Malekghassemi743c10c2015-06-16 18:05:27 -0700156
Masood Malekghassemi586e3832016-06-03 19:29:12 -0700157LDFLAGS = tuple(EXTRA_LINK_ARGS)
158CFLAGS = tuple(EXTRA_COMPILE_ARGS)
Craig Tiller4bef7ce2016-02-02 08:38:43 -0800159if "linux" in sys.platform or "darwin" in sys.platform:
Leifur Halldor Asgeirsson38a3d7a2016-03-01 10:00:08 -0500160 pymodinit_type = 'PyObject*' if PY3 else 'void'
Leifur Halldor Asgeirsson38a3d7a2016-03-01 10:00:08 -0500161 pymodinit = '__attribute__((visibility ("default"))) {}'.format(pymodinit_type)
162 DEFINE_MACROS += (('PyMODINIT_FUNC', pymodinit),)
Craig Tiller4bef7ce2016-02-02 08:38:43 -0800163
Ken Payson1efb6012016-06-08 13:06:44 -0700164# By default, Python3 distutils enforces compatibility of
165# c plugins (.so files) with the OSX version Python3 was built with.
166# For Python3.4, this is OSX 10.6, but we need Thread Local Support (__thread)
167if 'darwin' in sys.platform and PY3:
168 mac_target = sysconfig.get_config_var('MACOSX_DEPLOYMENT_TARGET')
169 if mac_target and (pkg_resources.parse_version(mac_target) <
170 pkg_resources.parse_version('10.7.0')):
171 os.environ['MACOSX_DEPLOYMENT_TARGET'] = '10.7'
Masood Malekghassemi398b06e2016-07-15 23:17:41 -0700172 os.environ['_PYTHON_HOST_PLATFORM'] = re.sub(
173 r'macosx-[0-9]+\.[0-9]+-(.+)',
174 r'macosx-10.7-\1',
175 util.get_platform())
Ken Payson1efb6012016-06-08 13:06:44 -0700176
Masood Malekghassemifd9cc102016-07-21 14:20:43 -0700177def cython_extensions_and_necessity():
178 cython_module_files = [os.path.join(PYTHON_STEM,
179 name.replace('.', '/') + '.pyx')
180 for name in CYTHON_EXTENSION_MODULE_NAMES]
Masood Malekghassemi5a65bcd2015-09-25 11:27:10 -0700181 extensions = [
182 _extension.Extension(
Masood Malekghassemi116982e2015-12-11 15:53:38 -0800183 name=module_name,
Masood Malekghassemifd9cc102016-07-21 14:20:43 -0700184 sources=[module_file] + list(CYTHON_HELPER_C_FILES) + list(CORE_C_FILES),
185 include_dirs=list(EXTENSION_INCLUDE_DIRECTORIES),
186 libraries=list(EXTENSION_LIBRARIES),
187 define_macros=list(DEFINE_MACROS),
Craig Tillera7f3e052016-02-05 15:02:39 -0800188 extra_compile_args=list(CFLAGS),
189 extra_link_args=list(LDFLAGS),
Masood Malekghassemifd9cc102016-07-21 14:20:43 -0700190 ) for (module_name, module_file) in zip(list(CYTHON_EXTENSION_MODULE_NAMES), cython_module_files)
Masood Malekghassemi5a65bcd2015-09-25 11:27:10 -0700191 ]
Masood Malekghassemifd9cc102016-07-21 14:20:43 -0700192 need_cython = BUILD_WITH_CYTHON
193 if not BUILD_WITH_CYTHON:
194 need_cython = need_cython or not commands.check_and_update_cythonization(extensions)
195 return commands.try_cythonize(extensions, linetracing=ENABLE_CYTHON_TRACING, mandatory=BUILD_WITH_CYTHON), need_cython
Masood Malekghassemi5a65bcd2015-09-25 11:27:10 -0700196
Masood Malekghassemifd9cc102016-07-21 14:20:43 -0700197CYTHON_EXTENSION_MODULES, need_cython = cython_extensions_and_necessity()
Masood Malekghassemi5a65bcd2015-09-25 11:27:10 -0700198
Masood Malekghassemi7566c9a2015-10-21 20:29:23 -0700199PACKAGE_DIRECTORIES = {
Masood Malekghassemi116982e2015-12-11 15:53:38 -0800200 '': PYTHON_STEM,
Nathaniel Manistab8b0adf2015-01-29 00:30:51 +0000201}
202
Masood Malekghassemi7566c9a2015-10-21 20:29:23 -0700203INSTALL_REQUIRES = (
Ken Payson2b7fae02016-06-28 17:20:37 -0700204 'six>=1.5.2',
Nathaniel Manistaa1880222015-09-10 22:16:04 +0000205 'enum34>=1.0.4',
Masood Malekghassemiad32ff42016-01-21 15:15:38 -0800206 # TODO(atash): eventually split the grpcio package into a metapackage
207 # depending on protobuf and the runtime component (independent of protobuf)
Ken Paysone5fd01a2016-08-02 12:06:21 -0700208 'protobuf>=3.0.0',
Masood Malekghassemid65632a2015-07-27 14:30:09 -0700209)
210
aaronjhengb273c042016-08-29 12:02:48 +0800211if not PY3:
212 INSTALL_REQUIRES += ('futures>=2.2.0',)
213
Ken Payson2b7fae02016-06-28 17:20:37 -0700214SETUP_REQUIRES = INSTALL_REQUIRES + (
Masood Malekghassemid65632a2015-07-27 14:30:09 -0700215 'sphinx>=1.3',
Ken Payson2b7fae02016-06-28 17:20:37 -0700216 'sphinx_rtd_theme>=0.1.8',
217 'six>=1.10',
Masood Malekghassemi2da46662016-10-25 15:57:48 -0700218 ) if ENABLE_DOCUMENTATION_BUILD else ()
219
Masood Malekghassemifd9cc102016-07-21 14:20:43 -0700220if BUILD_WITH_CYTHON:
221 sys.stderr.write(
222 "You requested a Cython build via GRPC_PYTHON_BUILD_WITH_CYTHON, "
223 "but do not have Cython installed. We won't stop you from using "
224 "other commands, but the extension files will fail to build.\n")
225elif need_cython:
226 sys.stderr.write(
227 'We could not find Cython. Setup may take 10-20 minutes.\n')
228 SETUP_REQUIRES += ('cython>=0.23',)
Masood Malekghassemid65632a2015-07-27 14:30:09 -0700229
Masood Malekghassemi7566c9a2015-10-21 20:29:23 -0700230COMMAND_CLASS = {
Masood Malekghassemi791cc312015-07-27 14:22:33 -0700231 'doc': commands.SphinxDocumentation,
Masood Malekghassemi5c147632015-07-31 14:08:19 -0700232 'build_project_metadata': commands.BuildProjectMetadata,
233 'build_py': commands.BuildPy,
Masood Malekghassemi1d177812016-01-12 09:21:57 -0800234 'build_ext': commands.BuildExt,
Masood Malekghassemi7566c9a2015-10-21 20:29:23 -0700235 'gather': commands.Gather,
Masood Malekghassemid65632a2015-07-27 14:30:09 -0700236}
237
Masood Malekghassemi6d2ef172016-01-04 15:33:17 -0800238# Ensure that package data is copied over before any commands have been run:
Masood Malekghassemi58a04942016-07-16 01:42:52 -0700239credentials_dir = os.path.join(PYTHON_STEM, 'grpc', '_cython', '_credentials')
Masood Malekghassemi6d2ef172016-01-04 15:33:17 -0800240try:
241 os.mkdir(credentials_dir)
242except OSError:
243 pass
Masood Malekghassemi58a04942016-07-16 01:42:52 -0700244shutil.copyfile(os.path.join('etc', 'roots.pem'),
245 os.path.join(credentials_dir, 'roots.pem'))
Masood Malekghassemi6d2ef172016-01-04 15:33:17 -0800246
Masood Malekghassemidf1e07e2016-02-02 14:17:14 -0800247PACKAGE_DATA = {
Masood Malekghassemif7474092016-02-12 12:04:53 -0800248 # Binaries that may or may not be present in the final installation, but are
249 # mentioned here for completeness.
Masood Malekghassemidf1e07e2016-02-02 14:17:14 -0800250 'grpc._cython': [
Nathaniel Manistafa6cad72016-02-05 21:26:54 +0000251 '_credentials/roots.pem',
Masood Malekghassemidf1e07e2016-02-02 14:17:14 -0800252 '_windows/grpc_c.32.python',
253 '_windows/grpc_c.64.python',
254 ],
255}
Masood Malekghassemi1ff429d2016-06-02 16:39:20 -0700256PACKAGES = setuptools.find_packages(PYTHON_STEM)
Masood Malekghassemi7566c9a2015-10-21 20:29:23 -0700257
Masood Malekghassemiab5309c2016-05-05 18:39:01 -0700258setuptools.setup(
259 name='grpcio',
260 version=grpc_version.VERSION,
261 license=LICENSE,
262 ext_modules=CYTHON_EXTENSION_MODULES,
263 packages=list(PACKAGES),
264 package_dir=PACKAGE_DIRECTORIES,
Masood Malekghassemiab5309c2016-05-05 18:39:01 -0700265 package_data=PACKAGE_DATA,
266 install_requires=INSTALL_REQUIRES,
267 setup_requires=SETUP_REQUIRES,
268 cmdclass=COMMAND_CLASS,
Masood Malekghassemiab5309c2016-05-05 18:39:01 -0700269)