Skip to content
Snippets Groups Projects
setup.py 4.45 KiB
Newer Older
# Based on https://github.com/pybind/cmake_example

import os
import re
import sys
import platform
import subprocess
Ian Bell's avatar
Ian Bell committed
import timeit

from setuptools import setup, Extension
from setuptools.command.build_ext import build_ext
from distutils.version import LooseVersion

# VERSION is now read from teqpversion.hpp header file
match = re.search(r'TEQPVERSION = \"([0-9a-z.]+)\"\;', open('interface/teqpversion.hpp').read())
if match:
    VERSION = match.group(1)
else:
    raise ValueError("Unable to parse version string from interface/teqpversion.hpp")

here = os.path.dirname(os.path.abspath(__file__))

Ian Bell's avatar
Ian Bell committed
tic = timeit.default_timer()

class CMakeExtension(Extension):
    def __init__(self, name, sourcedir=''):
        Extension.__init__(self, name, sources=[])
        self.sourcedir = os.path.abspath(sourcedir)

class CMakeBuild(build_ext):
    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,
                      '-DPYTHON_EXECUTABLE=' + sys.executable]

        cfg = 'Debug' if self.debug else 'Release'
        build_args = ['--config', cfg]

        if platform.system() == "Windows":
            cmake_args += ['-DCMAKE_LIBRARY_OUTPUT_DIRECTORY_{}={}'.format(cfg.upper(), extdir)]
            cmake_args += ['-T ClangCL']
            if sys.maxsize > 2**32:
                cmake_args += ['-A', 'x64']
            build_args += ['--', '/m']
        else:
            cmake_args += ['-DCMAKE_BUILD_TYPE=' + cfg]
            build_args += ['--', '-j2']

        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)
        
        # Config
        cmake_elements = ['cmake', ext.sourcedir] + cmake_args
        print('cmake config command:', ' '.join(cmake_elements))
        print('running from:', self.build_temp)
        subprocess.check_call(cmake_elements, cwd=self.build_temp, env=env)

        # Build
        build_elements = ['cmake', '--build', '.', '--target', 'teqp'] + build_args
        print('cmake build command:', ' '.join(build_elements))
        subprocess.check_call(build_elements, cwd=self.build_temp)

init_template  = r'''import os

# Bring all entities from the extension module into this namespace
from .teqp import * 

def get_datapath():
    """Get the absolute path to the folder containing the root of multi-fluid data"""
    return os.path.abspath(os.path.dirname(__file__)+"/fluiddata")
Ian Bell's avatar
Ian Bell committed

from .teqp import __version__
'''

def prepare():
    # Package up the fluid data files for the multi-fluid
    # model
    if os.path.exists('teqp'):
        shutil.rmtree('teqp')
    os.makedirs('teqp')
    shutil.copytree('mycp','teqp/fluiddata')

    # Make a temporary MANIFEST.in to avoid polluting the repository
    # since it only contains one line
    with open('MANIFEST.in','w') as fp:
        fp.write('recursive-include teqp *.json')
    
    with open('teqp/__init__.py', 'w') as fp:
        fp.write(init_template)

def teardown():
    shutil.rmtree('teqp')
    os.remove('MANIFEST.in')

try:
    prepare()
    setup(
        name='teqp',
        version=VERSION,
        author='Ian Bell and friends',
        author_email='ian.bell@nist.gov',
        description='Templated EQuation of state Package',
        long_description='',
        ext_modules=[CMakeExtension('teqp.teqp')], # teqp.teqp is the extension module that lives inside the teqp package
        packages=['teqp'],
        include_package_data=True,
        cmdclass=dict(build_ext=CMakeBuild),
        zip_safe=False,
    )
finally:
Ian Bell's avatar
Ian Bell committed
    teardown()

toc = timeit.default_timer()

print('elapsed:', toc-tic, 'seconds')