blob: a505fefe5114e4163dbea24f79d5ca2d9d67ead1 [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."""
Yuchen Zeng6694bb02017-01-23 17:09:51 -080031
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'),)
Yuchen Zengf64bf282016-08-11 21:21:32 -070055CARES_INCLUDE = (
Yuchen Zeng6694bb02017-01-23 17:09:51 -080056 os.path.join('third_party', 'cares'),
57 os.path.join('third_party', 'cares', 'cares'),)
Yuchen Zengf64bf282016-08-11 21:21:32 -070058if 'linux' in sys.platform:
Yuchen Zeng6694bb02017-01-23 17:09:51 -080059 CARES_INCLUDE += (os.path.join('third_party', 'cares', 'config_linux'),)
Yuchen Zengf64bf282016-08-11 21:21:32 -070060if 'darwin' in sys.platform:
Yuchen Zeng6694bb02017-01-23 17:09:51 -080061 CARES_INCLUDE += (os.path.join('third_party', 'cares', 'config_darwin'),)
Masood Malekghassemi116982e2015-12-11 15:53:38 -080062
Masood Malekghassemid65632a2015-07-27 14:30:09 -070063# Ensure we're in the proper directory whether or not we're being used by pip.
64os.chdir(os.path.dirname(os.path.abspath(__file__)))
Masood Malekghassemi58a1dc22016-01-21 14:23:55 -080065sys.path.insert(0, os.path.abspath(PYTHON_STEM))
Masood Malekghassemid65632a2015-07-27 14:30:09 -070066
Masood Malekghassemi116982e2015-12-11 15:53:38 -080067# Break import-style to ensure we can actually find our in-repo dependencies.
Yuchen Zeng6694bb02017-01-23 17:09:51 -080068import _unixccompiler_patch
Masood Malekghassemid65632a2015-07-27 14:30:09 -070069import commands
Masood Malekghassemi116982e2015-12-11 15:53:38 -080070import grpc_core_dependencies
Craig Tillerf008f062016-02-08 12:48:03 -080071import grpc_version
Masood Malekghassemi743c10c2015-06-16 18:05:27 -070072
Yuchen Zeng6694bb02017-01-23 17:09:51 -080073if 'win32' in sys.platform:
74 _unixccompiler_patch.monkeypatch_unix_compiler()
75
Masood Malekghassemi586e3832016-06-03 19:29:12 -070076
Masood Malekghassemidf97aec2016-01-11 16:19:14 -080077LICENSE = '3-clause BSD'
78
Masood Malekghassemi7566c9a2015-10-21 20:29:23 -070079# Environment variable to determine whether or not the Cython extension should
80# *use* Cython or use the generated C files. Note that this requires the C files
Masood Malekghassemif17f0f62016-07-08 12:07:49 -070081# to have been generated by building first *with* Cython support. Even if this
82# is set to false, if the script detects that the generated `.c` file isn't
83# present, then it will still attempt to use Cython.
Masood Malekghassemi7566c9a2015-10-21 20:29:23 -070084BUILD_WITH_CYTHON = os.environ.get('GRPC_PYTHON_BUILD_WITH_CYTHON', False)
Masood Malekghassemi743c10c2015-06-16 18:05:27 -070085
Masood Malekghassemi7566c9a2015-10-21 20:29:23 -070086# Environment variable to determine whether or not to enable coverage analysis
87# in Cython modules.
88ENABLE_CYTHON_TRACING = os.environ.get(
89 'GRPC_PYTHON_ENABLE_CYTHON_TRACING', False)
90
Masood Malekghassemi586e3832016-06-03 19:29:12 -070091# There are some situations (like on Windows) where CC, CFLAGS, and LDFLAGS are
92# entirely ignored/dropped/forgotten by distutils and its Cygwin/MinGW support.
93# We use these environment variables to thus get around that without locking
94# ourselves in w.r.t. the multitude of operating systems this ought to build on.
Masood Malekghassemid66be0b2016-07-07 11:24:39 -070095# We can also use these variables as a way to inject environment-specific
96# compiler/linker flags. We assume GCC-like compilers and/or MinGW as a
97# reasonable default.
98EXTRA_ENV_COMPILE_ARGS = os.environ.get('GRPC_PYTHON_CFLAGS', None)
99EXTRA_ENV_LINK_ARGS = os.environ.get('GRPC_PYTHON_LDFLAGS', None)
100if EXTRA_ENV_COMPILE_ARGS is None:
Yuchen Zeng6694bb02017-01-23 17:09:51 -0800101 EXTRA_ENV_COMPILE_ARGS = '-fno-wrapv'
102 if 'win32' in sys.platform:
Masood Malekghassemid66be0b2016-07-07 11:24:39 -0700103 # We use define flags here and don't directly add to DEFINE_MACROS below to
104 # ensure that the expert user/builder has a way of turning it off (via the
105 # envvars) without adding yet more GRPC-specific envvars.
106 # See https://sourceforge.net/p/mingw-w64/bugs/363/
107 if '32' in platform.architecture()[0]:
108 EXTRA_ENV_COMPILE_ARGS += ' -D_ftime=_ftime32 -D_timeb=__timeb32 -D_ftime_s=_ftime32_s'
109 else:
110 EXTRA_ENV_COMPILE_ARGS += ' -D_ftime=_ftime64 -D_timeb=__timeb64'
111 elif "linux" in sys.platform or "darwin" in sys.platform:
Yuchen Zeng6694bb02017-01-23 17:09:51 -0800112 EXTRA_ENV_COMPILE_ARGS += ' -fvisibility=hidden'
Masood Malekghassemid66be0b2016-07-07 11:24:39 -0700113if EXTRA_ENV_LINK_ARGS is None:
Yuchen Zeng6694bb02017-01-23 17:09:51 -0800114 EXTRA_ENV_LINK_ARGS = '-lpthread'
115 if 'win32' in sys.platform:
116 # TODO(atash) check if this is actually safe to just import and call on
117 # non-Windows (to avoid breaking import style)
118 from distutils.cygwinccompiler import get_msvcr
119 msvcr = get_msvcr()[0]
Masood Malekghassemid66be0b2016-07-07 11:24:39 -0700120 # TODO(atash) sift through the GCC specs to see if libstdc++ can have any
121 # influence on the linkage outcome on MinGW for non-C++ programs.
122 EXTRA_ENV_LINK_ARGS += (
123 ' -static-libgcc -static-libstdc++ -mcrtdll={msvcr} '
124 '-static'.format(msvcr=msvcr))
Yuchen Zeng6694bb02017-01-23 17:09:51 -0800125 elif "linux" in sys.platform:
Masood Malekghassemid66be0b2016-07-07 11:24:39 -0700126 EXTRA_ENV_LINK_ARGS += ' -Wl,-wrap,memcpy'
127EXTRA_COMPILE_ARGS = shlex.split(EXTRA_ENV_COMPILE_ARGS)
128EXTRA_LINK_ARGS = shlex.split(EXTRA_ENV_LINK_ARGS)
Masood Malekghassemi586e3832016-06-03 19:29:12 -0700129
Masood Malekghassemi7566c9a2015-10-21 20:29:23 -0700130CYTHON_EXTENSION_PACKAGE_NAMES = ()
Masood Malekghassemi5a65bcd2015-09-25 11:27:10 -0700131
Masood Malekghassemi116982e2015-12-11 15:53:38 -0800132CYTHON_EXTENSION_MODULE_NAMES = ('grpc._cython.cygrpc',)
Masood Malekghassemi5a65bcd2015-09-25 11:27:10 -0700133
Masood Malekghassemi3acddb22016-07-13 18:38:33 -0700134CYTHON_HELPER_C_FILES = ()
Masood Malekghassemidf1e07e2016-02-02 14:17:14 -0800135
Masood Malekghassemi586e3832016-06-03 19:29:12 -0700136CORE_C_FILES = tuple(grpc_core_dependencies.CORE_SOURCE_FILES)
Masood Malekghassemidf1e07e2016-02-02 14:17:14 -0800137
Masood Malekghassemi387e1162016-01-05 10:16:12 -0800138EXTENSION_INCLUDE_DIRECTORIES = (
Yuchen Zengf64bf282016-08-11 21:21:32 -0700139 (PYTHON_STEM,) + CORE_INCLUDE + BORINGSSL_INCLUDE + ZLIB_INCLUDE +
140 CARES_INCLUDE)
Nathaniel Manistaa4ead5d2015-01-31 01:33:27 +0000141
Masood Malekghassemi10509a22016-02-03 13:32:22 -0800142EXTENSION_LIBRARIES = ()
Masood Malekghassemic9c53ee2016-02-03 10:30:23 -0800143if "linux" in sys.platform:
Craig Tiller4bef7ce2016-02-02 08:38:43 -0800144 EXTENSION_LIBRARIES += ('rt',)
Masood Malekghassemi10509a22016-02-03 13:32:22 -0800145if not "win32" in sys.platform:
146 EXTENSION_LIBRARIES += ('m',)
Masood Malekghassemi586e3832016-06-03 19:29:12 -0700147if "win32" in sys.platform:
Yuchen Zeng6694bb02017-01-23 17:09:51 -0800148 EXTENSION_LIBRARIES += ('ws2_32',)
Nathaniel Manistaa4ead5d2015-01-31 01:33:27 +0000149
Masood Malekghassemi58a04942016-07-16 01:42:52 -0700150DEFINE_MACROS = (
151 ('OPENSSL_NO_ASM', 1), ('_WIN32_WINNT', 0x600),
Yuchen Zeng88d4e772016-08-16 22:34:02 -0700152 ('GPR_BACKWARDS_COMPATIBILITY_MODE', 1),)
Masood Malekghassemi586e3832016-06-03 19:29:12 -0700153if "win32" in sys.platform:
Yuchen Zeng6694bb02017-01-23 17:09:51 -0800154 DEFINE_MACROS += (('OPENSSL_WINDOWS', 1), ('WIN32_LEAN_AND_MEAN', 1),
155 ('CARES_STATICLIB', 1),)
Masood Malekghassemi586e3832016-06-03 19:29:12 -0700156 if '64bit' in platform.architecture()[0]:
157 DEFINE_MACROS += (('MS_WIN64', 1),)
Yuchen Zeng88d4e772016-08-16 22:34:02 -0700158else:
159 DEFINE_MACROS += (('HAVE_CONFIG_H', 1),)
Masood Malekghassemi743c10c2015-06-16 18:05:27 -0700160
Masood Malekghassemi586e3832016-06-03 19:29:12 -0700161LDFLAGS = tuple(EXTRA_LINK_ARGS)
162CFLAGS = tuple(EXTRA_COMPILE_ARGS)
Craig Tiller4bef7ce2016-02-02 08:38:43 -0800163if "linux" in sys.platform or "darwin" in sys.platform:
Leifur Halldor Asgeirsson38a3d7a2016-03-01 10:00:08 -0500164 pymodinit_type = 'PyObject*' if PY3 else 'void'
Leifur Halldor Asgeirsson38a3d7a2016-03-01 10:00:08 -0500165 pymodinit = '__attribute__((visibility ("default"))) {}'.format(pymodinit_type)
166 DEFINE_MACROS += (('PyMODINIT_FUNC', pymodinit),)
Craig Tiller4bef7ce2016-02-02 08:38:43 -0800167
Yuchen Zeng6694bb02017-01-23 17:09:51 -0800168
Ken Payson1efb6012016-06-08 13:06:44 -0700169# By default, Python3 distutils enforces compatibility of
170# c plugins (.so files) with the OSX version Python3 was built with.
171# For Python3.4, this is OSX 10.6, but we need Thread Local Support (__thread)
172if 'darwin' in sys.platform and PY3:
173 mac_target = sysconfig.get_config_var('MACOSX_DEPLOYMENT_TARGET')
174 if mac_target and (pkg_resources.parse_version(mac_target) <
175 pkg_resources.parse_version('10.7.0')):
176 os.environ['MACOSX_DEPLOYMENT_TARGET'] = '10.7'
Masood Malekghassemi398b06e2016-07-15 23:17:41 -0700177 os.environ['_PYTHON_HOST_PLATFORM'] = re.sub(
178 r'macosx-[0-9]+\.[0-9]+-(.+)',
179 r'macosx-10.7-\1',
180 util.get_platform())
Ken Payson1efb6012016-06-08 13:06:44 -0700181
Masood Malekghassemifd9cc102016-07-21 14:20:43 -0700182def cython_extensions_and_necessity():
183 cython_module_files = [os.path.join(PYTHON_STEM,
184 name.replace('.', '/') + '.pyx')
185 for name in CYTHON_EXTENSION_MODULE_NAMES]
Masood Malekghassemi5a65bcd2015-09-25 11:27:10 -0700186 extensions = [
187 _extension.Extension(
Masood Malekghassemi116982e2015-12-11 15:53:38 -0800188 name=module_name,
Masood Malekghassemifd9cc102016-07-21 14:20:43 -0700189 sources=[module_file] + list(CYTHON_HELPER_C_FILES) + list(CORE_C_FILES),
190 include_dirs=list(EXTENSION_INCLUDE_DIRECTORIES),
191 libraries=list(EXTENSION_LIBRARIES),
192 define_macros=list(DEFINE_MACROS),
Craig Tillera7f3e052016-02-05 15:02:39 -0800193 extra_compile_args=list(CFLAGS),
194 extra_link_args=list(LDFLAGS),
Masood Malekghassemifd9cc102016-07-21 14:20:43 -0700195 ) for (module_name, module_file) in zip(list(CYTHON_EXTENSION_MODULE_NAMES), cython_module_files)
Masood Malekghassemi5a65bcd2015-09-25 11:27:10 -0700196 ]
Masood Malekghassemifd9cc102016-07-21 14:20:43 -0700197 need_cython = BUILD_WITH_CYTHON
198 if not BUILD_WITH_CYTHON:
199 need_cython = need_cython or not commands.check_and_update_cythonization(extensions)
200 return commands.try_cythonize(extensions, linetracing=ENABLE_CYTHON_TRACING, mandatory=BUILD_WITH_CYTHON), need_cython
Masood Malekghassemi5a65bcd2015-09-25 11:27:10 -0700201
Masood Malekghassemifd9cc102016-07-21 14:20:43 -0700202CYTHON_EXTENSION_MODULES, need_cython = cython_extensions_and_necessity()
Masood Malekghassemi5a65bcd2015-09-25 11:27:10 -0700203
Masood Malekghassemi7566c9a2015-10-21 20:29:23 -0700204PACKAGE_DIRECTORIES = {
Masood Malekghassemi116982e2015-12-11 15:53:38 -0800205 '': PYTHON_STEM,
Nathaniel Manistab8b0adf2015-01-29 00:30:51 +0000206}
207
Masood Malekghassemi7566c9a2015-10-21 20:29:23 -0700208INSTALL_REQUIRES = (
Ken Payson2b7fae02016-06-28 17:20:37 -0700209 'six>=1.5.2',
Nathaniel Manistaa1880222015-09-10 22:16:04 +0000210 'enum34>=1.0.4',
Yuchen Zeng6694bb02017-01-23 17:09:51 -0800211 'futures>=2.2.0',
Masood Malekghassemiad32ff42016-01-21 15:15:38 -0800212 # TODO(atash): eventually split the grpcio package into a metapackage
213 # depending on protobuf and the runtime component (independent of protobuf)
Ken Paysone5fd01a2016-08-02 12:06:21 -0700214 'protobuf>=3.0.0',
Masood Malekghassemid65632a2015-07-27 14:30:09 -0700215)
216
Ken Payson2b7fae02016-06-28 17:20:37 -0700217SETUP_REQUIRES = INSTALL_REQUIRES + (
Masood Malekghassemid65632a2015-07-27 14:30:09 -0700218 'sphinx>=1.3',
Ken Payson2b7fae02016-06-28 17:20:37 -0700219 'sphinx_rtd_theme>=0.1.8',
220 'six>=1.10',
Yuchen Zeng6694bb02017-01-23 17:09:51 -0800221)
222if BUILD_WITH_CYTHON:
223 sys.stderr.write(
224 "You requested a Cython build via GRPC_PYTHON_BUILD_WITH_CYTHON, "
225 "but do not have Cython installed. We won't stop you from using "
226 "other commands, but the extension files will fail to build.\n")
227elif need_cython:
228 sys.stderr.write(
229 'We could not find Cython. Setup may take 10-20 minutes.\n')
230 SETUP_REQUIRES += ('cython>=0.23',)
Masood Malekghassemid65632a2015-07-27 14:30:09 -0700231
Masood Malekghassemi7566c9a2015-10-21 20:29:23 -0700232COMMAND_CLASS = {
Masood Malekghassemi791cc312015-07-27 14:22:33 -0700233 'doc': commands.SphinxDocumentation,
Masood Malekghassemi5c147632015-07-31 14:08:19 -0700234 'build_project_metadata': commands.BuildProjectMetadata,
235 'build_py': commands.BuildPy,
Masood Malekghassemi1d177812016-01-12 09:21:57 -0800236 'build_ext': commands.BuildExt,
Masood Malekghassemi7566c9a2015-10-21 20:29:23 -0700237 'gather': commands.Gather,
Masood Malekghassemid65632a2015-07-27 14:30:09 -0700238}
239
Masood Malekghassemi6d2ef172016-01-04 15:33:17 -0800240# Ensure that package data is copied over before any commands have been run:
Masood Malekghassemi58a04942016-07-16 01:42:52 -0700241credentials_dir = os.path.join(PYTHON_STEM, 'grpc', '_cython', '_credentials')
Masood Malekghassemi6d2ef172016-01-04 15:33:17 -0800242try:
243 os.mkdir(credentials_dir)
244except OSError:
245 pass
Masood Malekghassemi58a04942016-07-16 01:42:52 -0700246shutil.copyfile(os.path.join('etc', 'roots.pem'),
247 os.path.join(credentials_dir, 'roots.pem'))
Masood Malekghassemi6d2ef172016-01-04 15:33:17 -0800248
Masood Malekghassemidf1e07e2016-02-02 14:17:14 -0800249PACKAGE_DATA = {
Masood Malekghassemif7474092016-02-12 12:04:53 -0800250 # Binaries that may or may not be present in the final installation, but are
251 # mentioned here for completeness.
Masood Malekghassemidf1e07e2016-02-02 14:17:14 -0800252 'grpc._cython': [
Nathaniel Manistafa6cad72016-02-05 21:26:54 +0000253 '_credentials/roots.pem',
Masood Malekghassemidf1e07e2016-02-02 14:17:14 -0800254 '_windows/grpc_c.32.python',
255 '_windows/grpc_c.64.python',
256 ],
257}
Masood Malekghassemi1ff429d2016-06-02 16:39:20 -0700258PACKAGES = setuptools.find_packages(PYTHON_STEM)
Masood Malekghassemi7566c9a2015-10-21 20:29:23 -0700259
Masood Malekghassemiab5309c2016-05-05 18:39:01 -0700260setuptools.setup(
261 name='grpcio',
262 version=grpc_version.VERSION,
263 license=LICENSE,
264 ext_modules=CYTHON_EXTENSION_MODULES,
265 packages=list(PACKAGES),
266 package_dir=PACKAGE_DIRECTORIES,
Masood Malekghassemiab5309c2016-05-05 18:39:01 -0700267 package_data=PACKAGE_DATA,
268 install_requires=INSTALL_REQUIRES,
269 setup_requires=SETUP_REQUIRES,
270 cmdclass=COMMAND_CLASS,
Masood Malekghassemiab5309c2016-05-05 18:39:01 -0700271)