-
Notifications
You must be signed in to change notification settings - Fork 526
/
setup.py
366 lines (328 loc) · 13 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
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
# ___________________________________________________________________________
#
# Pyomo: Python Optimization Modeling Objects
# Copyright (c) 2008-2024
# National Technology and Engineering Solutions of Sandia, LLC
# Under the terms of Contract DE-NA0003525 with National Technology and
# Engineering Solutions of Sandia, LLC, the U.S. Government retains certain
# rights in this software.
# This software is distributed under the 3-clause BSD License.
# ___________________________________________________________________________
"""
Script to generate the installer for pyomo.
"""
import os
import platform
import sys
from setuptools import setup, find_packages, Command
try:
# This works beginning in setuptools 40.7.0 (27 Jan 2019)
from setuptools import DistutilsOptionError
except ImportError:
# Needed for setuptools prior to 40.7.0
from distutils.errors import DistutilsOptionError
def read(*rnames):
with open(os.path.join(os.path.dirname(__file__), *rnames)) as README:
# Strip all leading badges up to, but not including the COIN-OR
# badge so that they do not appear in the PyPI description
while True:
line = README.readline()
if 'COIN-OR' in line:
break
if line.strip() and '[![' not in line:
break
return line + README.read()
def import_pyomo_module(*path):
_module_globals = dict(globals())
_module_globals['__name__'] = None
_source = os.path.join(os.path.dirname(__file__), *path)
with open(_source) as _FILE:
exec(_FILE.read(), _module_globals)
return _module_globals
def get_version():
# Source pyomo/version/info.py to get the version number
return import_pyomo_module('pyomo', 'version', 'info.py')['__version__']
def check_config_arg(name):
if name in sys.argv:
sys.argv.remove(name)
return True
if name in os.getenv('PYOMO_SETUP_ARGS', '').split():
return True
return False
CYTHON_REQUIRED = "required"
if not any(
arg.startswith(cmd)
for cmd in ('build', 'install', 'bdist', 'wheel')
for arg in sys.argv
):
using_cython = False
elif sys.version_info[:2] < (3, 11):
using_cython = "automatic"
else:
using_cython = False
if check_config_arg('--with-cython'):
using_cython = CYTHON_REQUIRED
if check_config_arg('--without-cython'):
using_cython = False
ext_modules = []
if using_cython:
try:
if platform.python_implementation() != "CPython":
# break out of this try-except (disable Cython)
raise RuntimeError("Cython is only supported under CPython")
from Cython.Build import cythonize
#
# Note: The Cython developers recommend that you distribute C source
# files to users. But this is fine for evaluating the utility of Cython
#
import shutil
files = [
"pyomo/core/expr/numvalue.pyx",
"pyomo/core/expr/numeric_expr.pyx",
"pyomo/core/expr/logical_expr.pyx",
# "pyomo/core/expr/visitor.pyx",
"pyomo/core/util.pyx",
"pyomo/repn/standard_repn.pyx",
"pyomo/repn/plugins/cpxlp.pyx",
"pyomo/repn/plugins/gams_writer.pyx",
"pyomo/repn/plugins/baron_writer.pyx",
"pyomo/repn/plugins/ampl/ampl_.pyx",
]
for f in files:
shutil.copyfile(f[:-1], f)
ext_modules = cythonize(files, compiler_directives={"language_level": 3})
except:
if using_cython == CYTHON_REQUIRED:
print(
"""
ERROR: Cython was explicitly requested with --with-cython, but cythonization
of core Pyomo modules failed.
"""
)
raise
using_cython = False
if check_config_arg('--with-distributable-extensions'):
#
# Import the APPSI extension builder
# NOTE: There is inconsistent behavior in Windows for APPSI.
# As a result, we will NOT include these extensions in Windows.
if not sys.platform.startswith('win'):
appsi_extension = import_pyomo_module('pyomo', 'contrib', 'appsi', 'build.py')[
'get_appsi_extension'
](
in_setup=True,
appsi_root=os.path.join(
os.path.dirname(__file__), 'pyomo', 'contrib', 'appsi'
),
)
ext_modules.append(appsi_extension)
class DependenciesCommand(Command):
"""Custom setuptools command
This will output the list of dependencies, including any optional
dependencies for 'extras_require` targets. This is needed so that
we can (relatively) easily extract what `pip install '.[optional]'`
would have done so that we can pass it on to a 'conda install'
command when setting up Pyomo testing in a conda environment
(because conda for all intents does not acknowledge
`extras_require`).
"""
description = "list the dependencies for this package"
user_options = [('extras=', None, 'extra targets to include')]
def initialize_options(self):
self.extras = None
def finalize_options(self):
if self.extras is not None:
self.extras = [e for e in (_.strip() for _ in self.extras.split(',')) if e]
for e in self.extras:
if e not in setup_kwargs['extras_require']:
raise DistutilsOptionError(
"extras can only include {%s}"
% (', '.join(setup_kwargs['extras_require']))
)
def run(self):
deps = list(self._print_deps(setup_kwargs['install_requires']))
if self.extras is not None:
for e in self.extras:
deps.extend(self._print_deps(setup_kwargs['extras_require'][e]))
print(' '.join(deps))
def _print_deps(self, deplist):
class version_cmp(object):
ver = tuple(map(int, platform.python_version_tuple()[:2]))
def __lt__(self, other):
return self.ver < tuple(map(int, other.split('.')))
def __le__(self, other):
return self.ver <= tuple(map(int, other.split('.')))
def __gt__(self, other):
return not self.__le__(other)
def __ge__(self, other):
return not self.__lt__(other)
def __eq__(self, other):
return self.ver == tuple(map(int, other.split('.')))
def __ne__(self, other):
return not self.__eq__(other)
implementation_name = sys.implementation.name
platform_system = platform.system()
python_version = version_cmp()
for entry in deplist:
dep, _, condition = (_.strip() for _ in entry.partition(';'))
if condition and not eval(condition):
continue
yield dep
setup_kwargs = dict(
name='Pyomo',
#
# Note: the release number is set in pyomo/version/info.py
#
cmdclass={'dependencies': DependenciesCommand},
version=get_version(),
maintainer='Pyomo Developer Team',
maintainer_email='pyomo-developers@googlegroups.com',
url='http://pyomo.org',
project_urls={
'Documentation': 'https://pyomo.readthedocs.io/',
'Source': 'https://github.com/Pyomo/pyomo',
},
license='BSD',
platforms=["any"],
description='Pyomo: Python Optimization Modeling Objects',
long_description=read('README.md'),
long_description_content_type='text/markdown',
keywords=['optimization'],
classifiers=[
'Development Status :: 5 - Production/Stable',
'Intended Audience :: End Users/Desktop',
'Intended Audience :: Science/Research',
'License :: OSI Approved :: BSD License',
'Natural Language :: English',
'Operating System :: MacOS',
'Operating System :: Microsoft :: Windows',
'Operating System :: Unix',
'Programming Language :: Python',
'Programming Language :: Python :: 3',
'Programming Language :: Python :: 3.9',
'Programming Language :: Python :: 3.10',
'Programming Language :: Python :: 3.11',
'Programming Language :: Python :: 3.12',
'Programming Language :: Python :: 3.13',
'Programming Language :: Python :: Implementation :: CPython',
'Programming Language :: Python :: Implementation :: PyPy',
'Topic :: Scientific/Engineering :: Mathematics',
'Topic :: Software Development :: Libraries :: Python Modules',
],
python_requires='>=3.9',
install_requires=['ply'],
extras_require={
# There are certain tests that also require pytest-qt, but because those
# tests are so environment/machine specific, we are leaving these out of
# the dependencies.
'tests': ['coverage', 'parameterized', 'pybind11', 'pytest', 'pytest-parallel'],
'docs': [
'Sphinx>4',
'sphinx-copybutton',
'sphinx_rtd_theme>0.5',
'sphinxcontrib-jsmath',
'sphinxcontrib-napoleon',
'sphinx-toolbox>=2.16.0',
'sphinx-jinja2-compat>=0.1.1',
'numpy', # Needed by autodoc for pynumero
'scipy', # Needed by autodoc for pynumero
],
'optional': [
'dill', # No direct use, but improves lambda pickle
'ipython', # contrib.viewer
'linear-tree', # contrib.piecewise
# Note: matplotlib 3.6.1 has bug #24127, which breaks
# seaborn's histplot (triggering parmest failures)
# Note: minimum version from community_detection use of
# matplotlib.pyplot.get_cmap()
'matplotlib>=3.6.0,!=3.6.1',
# network, incidence_analysis, community_detection
# Note: networkx 3.2 is Python>-3.9, but there is a broken
# 3.2 package on conda-forge that will get implicitly
# installed on python 3.8
'networkx<3.2; python_version<"3.9"',
'networkx; python_version>="3.9"',
'numpy',
'openpyxl', # dataportals
#'pathos', # requested for #963, but PR currently closed
'pint', # units
'plotly', # incidence_analysis
'python-louvain', # community_detection
'pyyaml', # core
# qtconsole also requires a supported Qt version (PyQt5 or PySide6).
# Because those are environment specific, we have left that out here.
'qtconsole', # contrib.viewer
'scipy',
'sympy', # differentiation
'xlrd', # dataportals
'z3-solver', # community_detection
#
# subprocess output is merged more reliably if
# 'PeekNamedPipe' is available from pywin32
'pywin32; platform_system=="Windows"',
#
# The following optional dependencies are difficult to
# install on PyPy (binary wheels are not available), so we
# will only "require" them on other (CPython) platforms:
'casadi; implementation_name!="pypy"', # dae
'numdifftools; implementation_name!="pypy"', # pynumero
'pandas; implementation_name!="pypy"',
'seaborn; implementation_name!="pypy"', # parmest.graphics
],
},
packages=find_packages(exclude=("scripts",)),
package_data={
"pyomo.contrib.ampl_function_demo": ["src/*"],
"pyomo.contrib.appsi.cmodel": ["src/*"],
"pyomo.contrib.mcpp": ["*.cpp"],
"pyomo.contrib.pynumero": ['src/*', 'src/tests/*'],
"pyomo.contrib.viewer": ["*.ui"],
"pyomo.contrib.simplification.ginac": ["src/*.cpp", "src/*.hpp"],
},
ext_modules=ext_modules,
entry_points="""
[console_scripts]
pyomo = pyomo.scripting.pyomo_main:main_console_script
[pyomo.command]
pyomo.help = pyomo.scripting.driver_help
pyomo.viewer=pyomo.contrib.viewer.pyomo_viewer
""",
)
try:
setup(**setup_kwargs)
except SystemExit as e_info:
# Cython can generate a SystemExit exception on Windows if the
# environment is missing / has an incorrect Microsoft compiler.
# Since Cython is not strictly required, we will disable Cython and
# try re-running setup(), but only for this very specific situation.
if 'Microsoft Visual C++' not in str(e_info):
raise
elif using_cython == CYTHON_REQUIRED:
print(
"""
ERROR: Cython was explicitly requested with --with-cython, but cythonization
of core Pyomo modules failed.
"""
)
raise
else:
print(
"""
ERROR: setup() failed:
%s
Re-running setup() without the Cython modules
"""
% (str(e_info),)
)
setup_kwargs['ext_modules'] = []
setup(**setup_kwargs)
print(
"""
WARNING: Installation completed successfully, but the attempt to cythonize
core Pyomo modules failed. Cython provides performance
optimizations and is not required for any Pyomo functionality.
Cython returned the following error:
"%s"
"""
% (str(e_info),)
)