forked from BlueBrain/libsonata
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsetup.py
170 lines (137 loc) · 4.81 KB
/
setup.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
import inspect
import os
import platform
import re
import subprocess
import sys
from setuptools import setup, Extension
from setuptools.command.build_ext import build_ext
from setuptools.command.install import install
from setuptools.command.test import test
from distutils.version import LooseVersion
REQUIRED_NUMPY_VERSION = "numpy>=1.12.0"
MIN_CPU_CORES = 2
class lazy_dict(dict):
"""When the value associated to a key is a function, then returns
the function call instead of the function.
"""
def __getitem__(self, item):
value = dict.__getitem__(self, item)
if inspect.isfunction(value):
return value()
return value
def get_sphinx_command():
"""Lazy load of Sphinx distutils command class
"""
from sphinx.setup_command import BuildDoc
return BuildDoc
def get_cpu_count():
try:
return len(os.sched_getaffinity(0)) # linux only
except:
pass
try:
return os.cpu_count() # python 3.4+
except:
return 1 # default
class CMakeExtension(Extension):
def __init__(self, name, sourcedir=""):
Extension.__init__(self, name, sources=[])
self.sourcedir = os.path.abspath(sourcedir)
class CMakeBuild(build_ext, object):
user_options = build_ext.user_options + [
('target=', None, "specify the CMake target to build")
]
def initialize_options(self):
self.target = "sonata_python"
super(CMakeBuild, self).initialize_options()
def run(self):
try:
out = subprocess.check_output(["cmake", "--version"])
except OSError:
raise RuntimeError(
"CMake must be installed to build the following extensions: "
+ ", ".join(e.name for e in self.extensions)
)
if platform.system() == "Windows":
cmake_version = LooseVersion(
re.search(r"version\s*([\d.]+)", out.decode()).group(1)
)
if cmake_version < "3.1.0":
raise RuntimeError("CMake >= 3.1.0 is required on Windows")
for ext in self.extensions:
self.build_extension(ext)
def build_extension(self, ext):
extdir = os.path.abspath(os.path.dirname(self.get_ext_fullpath(ext.name)))
cmake_args = [
"-DCMAKE_LIBRARY_OUTPUT_DIRECTORY=" + extdir,
"-DEXTLIB_FROM_SUBMODULES=ON",
"-DSONATA_PYTHON=ON",
"-DSONATA_VERSION=" + self.distribution.get_version(),
"-DCMAKE_BUILD_TYPE=",
"-DSONATA_CXX_WARNINGS=OFF",
'-DPYTHON_EXECUTABLE=' + sys.executable
]
optimize = "OFF" if self.debug else "ON"
build_args = ["--config", optimize, "--target", self.target]
if platform.system() == "Windows":
cmake_args += [
"-DCMAKE_LIBRARY_OUTPUT_DIRECTORY_{}={}".format(
optimize.upper(), extdir
)
]
if sys.maxsize > 2 ** 32:
cmake_args += ["-A", "x64"]
build_args += ["--", "/m"]
else:
build_args += ["--", "-j{}".format(max(MIN_CPU_CORES, get_cpu_count()))]
env = os.environ.copy()
env["CXXFLAGS"] = '{} -DVERSION_INFO=\\"{}\\"'.format(
env.get("CXXFLAGS", ""), self.distribution.get_version()
)
if not os.path.exists(self.build_temp):
os.makedirs(self.build_temp)
subprocess.check_call(
["cmake", ext.sourcedir] + cmake_args, cwd=self.build_temp, env=env
)
subprocess.check_call(
["cmake", "--build", "."] + build_args, cwd=self.build_temp
)
class PkgTest(test):
"""Custom disutils command that acts like as a replacement
for the "test" command.
"""
new_commands = [('test_ext', lambda self: True), ('test_doc', lambda self: True)]
sub_commands = test.sub_commands + new_commands
def run(self):
super(PkgTest, self).run()
self.run_command('test_ext')
self.run_command('test_doc')
install_requires = [
REQUIRED_NUMPY_VERSION,
]
setup_requires = [
"setuptools_scm",
]
# HACK: `setup_requires` uses `easy_install` to do the 'pre-installation' of
# numpy, it doesn't realize it has to install numpy version that supports py2.
if sys.version_info[0] == 3:
setup_requires.append(REQUIRED_NUMPY_VERSION)
setup(
name="libsonata",
description='SONATA files reader',
author="BlueBrain Project, EPFL",
author_email="[email protected]",
classifiers=[],
ext_modules=[CMakeExtension("libsonata")],
cmdclass=lazy_dict(
build_ext=CMakeBuild,
test_ext=CMakeBuild,
test=PkgTest,
test_doc=get_sphinx_command,
),
zip_safe=False,
setup_requires=setup_requires,
install_requires=install_requires,
use_scm_version=True,
)