import os import sys import subprocess import datetime from setuptools import setup, Extension, find_packages from setuptools.command.build_ext import build_ext from distutils.version import StrictVersion import setup.helper as setup_helper from setup.config import SetupConfig if sys.version_info[0] == 2: sys.exit('Sorry, Python 2.x is not supported') # Minimal storm version required storm_min_version = "1.5.1" # Get the long description from the README file with open(os.path.join(os.path.abspath(os.path.dirname(__file__)), 'README.md'), encoding='utf-8') as f: long_description = f.read() class CMakeExtension(Extension): def __init__(self, name, sourcedir='', subdir=''): Extension.__init__(self, name, sources=[]) self.sourcedir = os.path.abspath(sourcedir) self.subdir = subdir class CMakeBuild(build_ext): user_options = build_ext.user_options + [ ('storm-dir=', None, 'Path to storm root (binary) location'), ('disable-dft', None, 'Disable support for DFTs'), ('disable-pars', None, 'Disable support for parametric models'), ('disable-pomdp', None, 'Disable support for POMDP analysis'), ('debug', None, 'Build in Debug mode'), ('jobs=', 'j', 'Number of jobs to use for compiling') ] config = SetupConfig() def _extdir(self, extname): return os.path.abspath(os.path.dirname(self.get_ext_fullpath(extname))) def run(self): try: _ = 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)) # Build cmake version info print("Stormpy - Building into {}".format(self.build_temp)) build_temp_version = self.build_temp + "-version" setup_helper.ensure_dir_exists(build_temp_version) # Write config setup_helper.ensure_dir_exists(self.build_temp) self.config.write_config(os.path.join(self.build_temp, "build_config.cfg")) cmake_args = [] storm_dir = os.path.expanduser(self.config.get_as_string("storm_dir")) if storm_dir: cmake_args += ['-Dstorm_DIR=' + storm_dir] _ = subprocess.check_output(['cmake', os.path.abspath("cmake")] + cmake_args, cwd=build_temp_version) cmake_conf = setup_helper.load_cmake_config(os.path.join(build_temp_version, 'generated/config.py')) # Set storm directory if storm_dir == "": storm_dir = cmake_conf.STORM_DIR if storm_dir != cmake_conf.STORM_DIR: print("Stormpy - Warning: Using different storm directory {} instead of given {}!".format( cmake_conf.STORM_DIR, storm_dir)) storm_dir = cmake_conf.STORM_DIR # Check version storm_version, storm_commit = setup_helper.parse_storm_version(cmake_conf.STORM_VERSION) if StrictVersion(storm_version) < StrictVersion(storm_min_version): sys.exit( 'Stormpy - Error: Storm version {} from \'{}\' is not supported anymore!'.format(storm_version, storm_dir)) # Check additional support use_dft = cmake_conf.HAVE_STORM_DFT and not self.config.get_as_bool("disable_dft") use_pars = cmake_conf.HAVE_STORM_PARS and not self.config.get_as_bool("disable_pars") use_pomdp = cmake_conf.HAVE_STORM_POMDP #and not self.config.get_as_bool("disable_pomdp") # Print build info print("Stormpy - Using storm {} from {}".format(storm_version, storm_dir)) if use_dft: print("Stormpy - Support for DFTs found and included.") else: print("Stormpy - Warning: No support for DFTs!") if use_pars: print("Stormpy - Support for parametric models found and included.") else: print("Stormpy - Warning: No support for parametric models!") if use_pomdp: print("Stormpy - Support for POMDP analysis found and included.") else: print("Stormpy - Warning: No support for POMDP analysis!") # Set general cmake build options build_type = 'Debug' if self.config.get_as_bool("debug") else 'Release' cmake_args = ['-DPYTHON_EXECUTABLE=' + sys.executable] cmake_args += ['-DCMAKE_BUILD_TYPE=' + build_type] if storm_dir is not None: cmake_args += ['-Dstorm_DIR=' + storm_dir] if use_dft: cmake_args += ['-DHAVE_STORM_DFT=ON'] if use_pars: cmake_args += ['-DHAVE_STORM_PARS=ON'] if use_pomdp: cmake_args += ['-DHAVE_STORM_POMDP=ON'] build_args = ['--config', build_type] build_args += ['--', '-j{}'.format(self.config.get_as_int("jobs"))] # Build extensions for ext in self.extensions: setup_helper.ensure_dir_exists(os.path.join(self._extdir(ext.name), ext.subdir)) if ext.name == "core": with open(os.path.join(self._extdir(ext.name), ext.subdir, "_config.py"), "w") as f: f.write("# Generated from setup.py at {}\n".format(datetime.datetime.now())) f.write("import pycarl\n") if cmake_conf.STORM_CLN_EA or cmake_conf.STORM_CLN_RF: f.write("import pycarl.cln\n") if not cmake_conf.STORM_CLN_EA or not cmake_conf.STORM_CLN_RF: f.write("import pycarl.gmp\n") if cmake_conf.STORM_CLN_EA: f.write("Rational = pycarl.cln.Rational\n") else: f.write("Rational = pycarl.gmp.Rational\n") if cmake_conf.STORM_CLN_RF: rfpackage = "cln" else: rfpackage = "gmp" f.write("RationalRF = pycarl.{}.Rational\n".format(rfpackage)) f.write("Polynomial = pycarl.{}.Polynomial\n".format(rfpackage)) f.write("FactorizedPolynomial = pycarl.{}.FactorizedPolynomial\n".format(rfpackage)) f.write("RationalFunction = pycarl.{}.RationalFunction\n".format(rfpackage)) f.write("FactorizedRationalFunction = pycarl.{}.FactorizedRationalFunction\n".format(rfpackage)) f.write("\n") f.write("storm_with_dft = {}\n".format(use_dft)) f.write("storm_with_pars = {}\n".format(use_pars)) elif ext.name == "info": with open(os.path.join(self._extdir(ext.name), ext.subdir, "_config.py"), "w") as f: f.write("# Generated from setup.py at {}\n".format(datetime.datetime.now())) f.write("storm_version = \"{}\"\n".format(storm_version)) f.write("storm_cln_ea = {}\n".format(cmake_conf.STORM_CLN_EA)) f.write("storm_cln_rf = {}".format(cmake_conf.STORM_CLN_RF)) elif ext.name == "dft": with open(os.path.join(self._extdir(ext.name), ext.subdir, "_config.py"), "w") as f: f.write("# Generated from setup.py at {}\n".format(datetime.datetime.now())) f.write("storm_with_dft = {}".format(use_dft)) if not use_dft: print("Stormpy - DFT bindings skipped") continue elif ext.name == "pars": with open(os.path.join(self._extdir(ext.name), ext.subdir, "_config.py"), "w") as f: f.write("# Generated from setup.py at {}\n".format(datetime.datetime.now())) f.write("storm_with_pars = {}".format(use_pars)) if not use_pars: print("Stormpy - Bindings for parametric models skipped") continue elif ext.name == "pomdp": with open(os.path.join(self._extdir(ext.name), ext.subdir, "_config.py"), "w") as f: f.write("# Generated from setup.py at {}\n".format(datetime.datetime.now())) f.write("storm_with_pomdp = {}".format(use_pomdp)) if not use_pomdp: print("Stormpy - Bindings for POMDP analysis skipped") continue self.build_extension(ext, cmake_args, build_args) def initialize_options(self): build_ext.initialize_options(self) # Set default values for custom cmdline flags self.storm_dir = None self.disable_dft = None self.disable_pars = None self.disable_pomdp = None self.debug = None self.jobs = None def finalize_options(self): build_ext.finalize_options(self) # Load setup config # This can only be done after the finalization step, because otherwise build_temp is not initialized yet. self.config.load_from_file(os.path.join(self.build_temp, "build_config.cfg")) # Update setup config self.config.update("storm_dir", self.storm_dir) self.config.update("disable_dft", self.disable_dft) self.config.update("disable_pars", self.disable_pars) self.config.update("disable_pomdp", self.disable_pomdp) self.config.update("debug", self.debug) self.config.update("jobs", self.jobs) def build_extension(self, ext, general_cmake_args, general_build_args): extdir = os.path.abspath(os.path.dirname(self.get_ext_fullpath(ext.name))) cmake_args = ['-DCMAKE_LIBRARY_OUTPUT_DIRECTORY=' + os.path.join(extdir, ext.subdir)] + general_cmake_args build_args = general_build_args env = os.environ.copy() env['CXXFLAGS'] = '{} -DVERSION_INFO=\\"{}\\"'.format(env.get('CXXFLAGS', ''), self.distribution.get_version()) setup_helper.ensure_dir_exists(self.build_temp) print("Stormpy - CMake args={}".format(cmake_args)) # Call cmake subprocess.check_call(['cmake', ext.sourcedir] + cmake_args, cwd=self.build_temp, env=env) subprocess.check_call(['cmake', '--build', '.', '--target', ext.name] + build_args, cwd=self.build_temp) setup( name="stormpy", version=setup_helper.obtain_version(), author="M. Volk", author_email="matthias.volk@cs.rwth-aachen.de", maintainer="S. Junges", maintainer_email="sebastian.junges@cs.rwth-aachen.de", url="https://github.com/moves-rwth/stormpy/", description="stormpy - Python Bindings for Storm", long_description=long_description, long_description_content_type='text/markdown', project_urls={ 'Documentation': 'https://moves-rwth.github.io/stormpy/', 'Source': 'https://github.com/moves-rwth/stormpy/', 'Bug reports': 'https://github.com/moves-rwth/stormpy/issues', }, classifiers=[ 'Intended Audience :: Science/Research', 'Topic :: Scientific/Engineering', 'Topic :: Software Development :: Libraries :: Python Modules', ], packages=find_packages('lib'), package_dir={'': 'lib'}, include_package_data=True, package_data={'stormpy.examples': ['examples/files/*']}, ext_package='stormpy', ext_modules=[CMakeExtension('core', subdir=''), CMakeExtension('info', subdir='info'), CMakeExtension('logic', subdir='logic'), CMakeExtension('storage', subdir='storage'), CMakeExtension('utility', subdir='utility'), CMakeExtension('dft', subdir='dft'), CMakeExtension('pars', subdir='pars'), CMakeExtension('pomdp', subdir='pomdp')], cmdclass={'build_ext': CMakeBuild}, zip_safe=False, install_requires=['pycarl>=2.0.4'], setup_requires=['pytest-runner'], tests_require=['pytest'], python_requires='>=3', )