-
Notifications
You must be signed in to change notification settings - Fork 33
/
setup.py
257 lines (217 loc) · 7.98 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
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
# SPDX-FileCopyrightText: Contributors to the Power Grid Model project <[email protected]>
#
# SPDX-License-Identifier: MPL-2.0
import os
import platform
import shutil
from itertools import chain
from pathlib import Path
# noinspection PyPackageRequirements
from setuptools import Extension, setup
from setuptools.command.build_ext import build_ext
from wheel.bdist_wheel import bdist_wheel
# determine platform, only windows or linux
if platform.system() == "Windows":
if_win = True
elif platform.system() in ["Linux", "Darwin"]:
if_win = False
else:
raise SystemError("Only Windows, Linux, or MacOS is supported!")
def get_required_dependency_include() -> list[str]:
"""
Get build requirements includes.
Returns:
either empty list or a list of header path
"""
try:
import libboost_headers
import msgpack_cxx
import nlohmann_json
return [str(msgpack_cxx.get_include()), str(nlohmann_json.get_include()), str(libboost_headers.get_include())]
except ImportError:
return []
def get_pre_installed_header_include() -> list[str]:
"""
Get header files from pybuild_header_dependency, if it is installed
Returns:
either empty list or a list of header path
"""
try:
from pybuild_header_dependency import HeaderResolver
resolver = HeaderResolver({"eigen": None})
return [str(resolver.get_include())]
except ImportError:
return []
def get_conda_include() -> list[str]:
"""
Get conda include path, if we are inside conda environment
Returns:
either empty list or a list of header paths
"""
include_paths = []
# in the conda build system the system root is defined in CONDA_PREFIX or BUILD_PREFIX
for prefix in ["CONDA_PREFIX", "BUILD_PREFIX"]:
if prefix in os.environ:
conda_path = os.environ[prefix]
if if_win:
# windows has Library folder prefix
include_paths.append(os.path.join(conda_path, "Library", "include"))
include_paths.append(os.path.join(conda_path, "Library", "include", "eigen3"))
else:
include_paths.append(os.path.join(conda_path, "include"))
include_paths.append(os.path.join(conda_path, "include", "eigen3"))
return include_paths
# custom class for ctypes
class CTypesExtension(Extension):
pass
class bdist_wheel_abi_none(bdist_wheel):
def finalize_options(self):
bdist_wheel.finalize_options(self)
self.root_is_pure = False
def get_tag(self):
python, abi, plat = bdist_wheel.get_tag(self)
return "py3", "none", plat
# custom compiler for linux
class MyBuildExt(build_ext):
def build_extensions(self):
if not if_win:
if "CXX" in os.environ:
cxx = os.environ["CXX"]
else:
cxx = self.compiler.compiler_cxx[0]
# check setuptools has an update change in the version 72.2 about cxx compiler options
# to be compatible with both version, we check if compiler_so_cxx exists
if not hasattr(self.compiler, "compiler_so_cxx"):
compiler_so_cxx = self.compiler.compiler_so
linker_so_cxx = self.compiler.linker_so
else:
compiler_so_cxx = self.compiler.compiler_so_cxx
linker_so_cxx = self.compiler.linker_so_cxx
# customize compiler and linker options
compiler_so_cxx[0] = cxx
linker_so_cxx[0] = cxx
self.compiler.compiler_cxx = [cxx]
# add link time optimization
if "clang" in cxx:
lto_flag = "-flto=thin"
else:
lto_flag = "-flto"
compiler_so_cxx += [lto_flag]
linker_so_cxx += [lto_flag]
# remove debug and optimization flags
for x in compiler_so_cxx.copy():
if x in ["-g", "-O2"]:
compiler_so_cxx.remove(x)
for x in linker_so_cxx.copy():
if x in ["-g", "-O2", "-Wl,-O1"]:
linker_so_cxx.remove(x)
print("-------compiler arguments----------")
print(compiler_so_cxx)
print("-------linker arguments----------")
print(linker_so_cxx)
return super().build_extensions()
def get_export_symbols(self, ext):
return ext.export_symbols
def get_ext_filename(self, ext_name):
return os.path.join(*ext_name.split(".")) + (".dll" if if_win else ".so")
def generate_build_ext(pkg_dir: Path, pkg_name: str):
"""
Generate extension dict for setup.py
the return value ext_dict, can be called in setup(**ext_dict)
Args:
pkg_dir:
pkg_name:
Returns:
"""
# fetch dependent headers
pgm = Path("power_grid_model")
pgm_c = Path("power_grid_model_c")
# include-folders
include_dirs = [
str(pkg_dir / pgm_c / pgm / "include"), # The include-folder of the library
str(pkg_dir / pgm_c / pgm_c / "include"), # The include-folder of the C API self
]
include_dirs += get_required_dependency_include()
include_dirs += get_pre_installed_header_include()
include_dirs += get_conda_include()
# compiler and link flag
cflags: list[str] = []
lflags: list[str] = []
library_dirs: list[str] = []
libraries: list[str] = []
sources = [
str(pgm_c / pgm_c / "src" / "handle.cpp"),
str(pgm_c / pgm_c / "src" / "meta_data.cpp"),
str(pgm_c / pgm_c / "src" / "model.cpp"),
str(pgm_c / pgm_c / "src" / "options.cpp"),
str(pgm_c / pgm_c / "src" / "dataset.cpp"),
str(pgm_c / pgm_c / "src" / "serialization.cpp"),
]
# macro
define_macros = [
("EIGEN_MPL2_ONLY", "1"), # only MPL-2 part of eigen3
]
pkg_bin_dir = pkg_dir / "src" / pkg_name
# remove old extension build
build_dir = pkg_dir / "build"
if build_dir.exists():
shutil.rmtree(build_dir)
# remove binary
bin_files = list(chain(pkg_bin_dir.rglob("*.so"), pkg_bin_dir.rglob("*.dll")))
for bin_file in bin_files:
print(f"Remove binary file: {bin_file}")
bin_file.unlink()
# build steps for Windows and Linux
# different treat for windows and linux
# determine platform specific options
if if_win:
# flag for C++20
cflags += ["/std:c++20"]
else:
# flags for Linux and Mac
cflags += ["-std=c++20", "-O3", "-fvisibility=hidden"]
lflags += ["-lpthread", "-O3"]
# extra flag for Mac
if platform.system() == "Darwin":
# compiler flag to set version
cflags.append("-mmacosx-version-min=10.15")
# list of extensions
exts = [
CTypesExtension(
name="power_grid_model._core._power_grid_core",
sources=sources,
include_dirs=include_dirs,
library_dirs=library_dirs,
libraries=libraries,
extra_compile_args=cflags,
extra_link_args=lflags,
define_macros=define_macros,
language="c++",
)
]
# return dict of exts
return dict(ext_modules=exts, cmdclass={"build_ext": MyBuildExt, "bdist_wheel": bdist_wheel_abi_none})
def set_version(pkg_dir: Path):
# if PYPI_VERSION does not exist, copy from VERSION
pypi_file = pkg_dir / "PYPI_VERSION"
if not pypi_file.exists():
with open(pkg_dir / "VERSION") as f:
version = f.read().strip().strip("\n")
with open(pypi_file, "w") as f:
f.write(version)
def prepare_pkg(setup_file: Path) -> dict:
"""
Args:
setup_file:
Returns:
"""
print(f"Build wheel from {setup_file}")
pkg_dir = setup_file.parent
# package description
pkg_pip_name = "power-grid-model"
pkg_name = pkg_pip_name.replace("-", "_")
set_version(pkg_dir)
return generate_build_ext(pkg_dir=pkg_dir, pkg_name=pkg_name)
setup(
**prepare_pkg(setup_file=Path(__file__).resolve()),
)