mirror of
https://github.com/CoolProp/CoolProp.git
synced 2026-01-15 00:48:18 -05:00
333 lines
13 KiB
Python
333 lines
13 KiB
Python
from __future__ import print_function
|
|
import platform
|
|
import subprocess, shutil, os, sys, glob
|
|
|
|
def copy_files():
|
|
def copytree(old,new):
|
|
print(old,'-->',new)
|
|
shutil.copytree(old, new)
|
|
def copy2(old, new):
|
|
print(old,'-->',new)
|
|
shutil.copy2(old, new)
|
|
|
|
import shutil
|
|
shutil.rmtree(os.path.join('CoolProp','include'), ignore_errors = True)
|
|
copytree(os.path.join(CProot, 'include'), os.path.join('CoolProp','include'))
|
|
for jsonfile in glob.glob(os.path.join('CoolProp','include','*_JSON.h')):
|
|
print('removing', jsonfile)
|
|
os.remove(jsonfile)
|
|
copytree(os.path.join(CProot, 'externals/cppformat/cppformat'), os.path.join('CoolProp','include','externals','cppformat','cppformat'))
|
|
copy2(os.path.join(CProot, 'CoolPropBibTeXLibrary.bib'), os.path.join('CoolProp', 'CoolPropBibTeXLibrary.bib'))
|
|
print('files copied.')
|
|
|
|
def remove_files():
|
|
import shutil
|
|
shutil.rmtree(os.path.join('CoolProp','include'), ignore_errors = True)
|
|
os.remove(os.path.join('CoolProp', 'CoolPropBibTeXLibrary.bib'))
|
|
print('files removed.')
|
|
|
|
def touch(fname):
|
|
open(fname, 'a').close()
|
|
os.utime(fname, None)
|
|
|
|
def recursive_collect_includes():
|
|
thefiles = []
|
|
include_path = os.path.join('CoolProp','include')
|
|
for root, dirs, files in os.walk(include_path):
|
|
thefiles += [os.path.relpath(os.path.join(root,_f), 'CoolProp') for _f in files]
|
|
return thefiles
|
|
|
|
if __name__=='__main__':
|
|
|
|
# Trying to change the standard library for C++
|
|
import platform
|
|
try:
|
|
macVersion = platform.mac_ver()[0].split('.')
|
|
if int(macVersion[0]) >= 10 and int(macVersion[1]) > 8:
|
|
os.environ["CC"] = "gcc"
|
|
os.environ["CXX"] = "g++"
|
|
print('switching compiler to g++ for OSX')
|
|
except:
|
|
pass
|
|
|
|
|
|
|
|
# ******************************
|
|
# CMAKE OPTIONS
|
|
# ******************************
|
|
|
|
# Example using CMake to build static library:
|
|
# python setup.py install --cmake-compiler vc9 --cmake-bitness 64
|
|
#
|
|
# or (because pip needs help)
|
|
#
|
|
# python setup.py install cmake=default,64
|
|
|
|
cmake_args = [_ for _ in sys.argv if _.startswith('cmake=')]
|
|
if cmake_args:
|
|
i = sys.argv.index(cmake_args[0])
|
|
sys.argv.pop(i)
|
|
cmake_compiler, cmake_bitness = cmake_args[0].split('cmake=')[1].split(',')
|
|
else:
|
|
if '--cmake-compiler' in sys.argv:
|
|
i = sys.argv.index('--cmake-compiler')
|
|
sys.argv.pop(i)
|
|
cmake_compiler = sys.argv.pop(i)
|
|
else:
|
|
cmake_compiler = ''
|
|
|
|
if '--cmake-bitness' in sys.argv:
|
|
i = sys.argv.index('--cmake-bitness')
|
|
sys.argv.pop(i)
|
|
cmake_bitness = sys.argv.pop(i)
|
|
else:
|
|
cmake_bitness = ''
|
|
|
|
USING_CMAKE = cmake_compiler or cmake_bitness
|
|
|
|
cmake_config_args = []
|
|
|
|
STATIC_LIBRARY_BUILT = False
|
|
if USING_CMAKE:
|
|
|
|
# Always force build since any changes in the C++ files will not force a rebuild
|
|
touch('CoolProp/CoolProp.pyx')
|
|
|
|
if 'clean' in sys.argv:
|
|
if os.path.exists('cmake_build'):
|
|
print('removing cmake_build folder...')
|
|
shutil.rmtree('cmake_build')
|
|
print('removed.')
|
|
|
|
cmake_config_args, cmake_build_args = [], []
|
|
if cmake_compiler == 'vc9':
|
|
cmake_build_args = ['--config','"Release"']
|
|
if cmake_bitness == '32':
|
|
cmake_config_args += ['-G','"Visual Studio 9 2008"']
|
|
elif cmake_bitness == '64':
|
|
cmake_config_args += ['-G','"Visual Studio 9 2008 Win64"']
|
|
else:
|
|
raise ValueError('cmake_bitness must be either 32 or 64; got ' + cmake_bitness)
|
|
elif cmake_compiler == 'vc10':
|
|
cmake_build_args = ['--config','"Release"']
|
|
if cmake_bitness == '32':
|
|
cmake_config_args += ['-G','"Visual Studio 10 2010"']
|
|
elif cmake_bitness == '64':
|
|
cmake_config_args += ['-G','"Visual Studio 10 2010 Win64"']
|
|
else:
|
|
raise ValueError('cmake_bitness must be either 32 or 64; got ' + cmake_bitness)
|
|
elif cmake_compiler == 'mingw':
|
|
cmake_config_args = ['-G','"MinGW Makefiles"']
|
|
if cmake_bitness == '32':
|
|
cmake_config_args += ['-DFORCE_BITNESS_32=ON']
|
|
elif cmake_bitness == '64':
|
|
cmake_config_args += ['-DFORCE_BITNESS_64=ON']
|
|
else:
|
|
raise ValueError('cmake_bitness must be either 32 or 64; got ' + cmake_bitness)
|
|
elif cmake_compiler == 'default':
|
|
cmake_config_args = []
|
|
if cmake_bitness == '32':
|
|
cmake_config_args += ['-DFORCE_BITNESS_32=ON']
|
|
elif cmake_bitness == '64':
|
|
cmake_config_args += ['-DFORCE_BITNESS_64=ON']
|
|
else:
|
|
raise ValueError('cmake_bitness must be either 32 or 64; got ' + cmake_bitness)
|
|
else:
|
|
raise ValueError('cmake_compiler [' + cmake_compiler + '] is invalid')
|
|
|
|
if 'darwin' in sys.platform:
|
|
cmake_config_args += ['-DCOOLPROP_OSX_105_COMPATIBILITY=ON']
|
|
if 'linux' in sys.platform:
|
|
cmake_config_args += ['-DCOOLPROP_FPIC=ON']
|
|
|
|
cmake_build_dir = os.path.join('cmake_build', '{compiler}-{bitness}bit'.format(compiler=cmake_compiler, bitness=cmake_bitness))
|
|
if not os.path.exists(cmake_build_dir):
|
|
os.makedirs(cmake_build_dir)
|
|
|
|
cmake_call_string = ' '.join(['cmake','../../../..','-DCOOLPROP_STATIC_LIBRARY=ON','-DCMAKE_VERBOSE_MAKEFILE=ON','-DCMAKE_BUILD_TYPE=Release'] + cmake_config_args)
|
|
print('calling: ' + cmake_call_string)
|
|
subprocess.check_call(cmake_call_string, shell = True, stdout = sys.stdout, stderr = sys.stderr, cwd = cmake_build_dir)
|
|
|
|
cmake_build_string = ' '.join(['cmake','--build', '.'] + cmake_build_args)
|
|
print('calling: ' + cmake_build_string)
|
|
subprocess.check_call(cmake_build_string, shell = True, stdout = sys.stdout, stderr = sys.stderr, cwd = cmake_build_dir)
|
|
|
|
# Now find the static library that we just built
|
|
static_libs = []
|
|
for search_suffix in ['Release/*.lib','Release/*.a', 'Debug/*.lib', 'Debug/*.a','*.a']:
|
|
static_libs += glob.glob(os.path.join(cmake_build_dir,search_suffix))
|
|
|
|
if len(static_libs) != 1:
|
|
raise ValueError("Found more than one static library using CMake build. Found: "+str(static_libs))
|
|
else:
|
|
STATIC_LIBRARY_BUILT = True
|
|
static_library_path = os.path.dirname(static_libs[0])
|
|
|
|
# Check if a sdist build for pypi
|
|
pypi = os.path.exists('.use_this_directory_as_root')
|
|
|
|
"""
|
|
Modes of operation:
|
|
1) Building the source distro (generate_headers.py must have been run before making the repo)
|
|
2) Installing from source (generate_headers.py must have been run before making the repo)
|
|
3) Installing from git repo (need to make sure to run generate_headers.py)
|
|
4)
|
|
"""
|
|
|
|
# Determine whether or not to use Cython - default is to use cython unless the file .build_without_cython is found in the current working directory
|
|
USE_CYTHON = not os.path.exists('.build_without_cython')
|
|
cy_ext = 'pyx' if USE_CYTHON else 'cpp'
|
|
|
|
if USE_CYTHON:
|
|
# Check for cython >= 0.21 due to the use of cpdef enum
|
|
try:
|
|
import Cython
|
|
except ImportError:
|
|
raise ImportError("Cython not found, please install it. You can do a pip install Cython")
|
|
|
|
from pkg_resources import parse_version
|
|
if parse_version(Cython.__version__) < parse_version('0.20'):
|
|
raise ImportError('Your version of Cython (%s) must be >= 0.20 . Please update your version of cython' % (Cython.__version__,))
|
|
|
|
if parse_version(Cython.__version__) >= parse_version('0.20'):
|
|
_profiling_enabled = True
|
|
else:
|
|
_profiling_enabled = False
|
|
|
|
if _profiling_enabled:
|
|
cython_directives = dict(profile = True,
|
|
embed_signature = True)
|
|
else:
|
|
cython_directives = dict(embed_signature = True)
|
|
else:
|
|
cython_directives = {}
|
|
|
|
# Determine the path to the root of the repository, the folder that contains the CMakeLists.txt file
|
|
# for normal builds, or the main directory for sdist builds
|
|
if pypi:
|
|
CProot = '.'
|
|
else:
|
|
if os.path.exists(os.path.join('..','..','CMakeLists.txt')):
|
|
# Good working directory
|
|
CProot = os.path.join('..','..')
|
|
else:
|
|
raise ValueError('Could not run script from this folder(' + os.path.abspath(os.path.curdir) + '). Run from wrappers/Python folder')
|
|
|
|
sys.path.append(os.path.join(CProot, 'dev'))
|
|
if not USING_CMAKE:
|
|
import generate_headers
|
|
# Generate the headers - does nothing if up to date - but only if not pypi
|
|
generate_headers.generate()
|
|
del generate_headers
|
|
|
|
import generate_constants_module
|
|
generate_constants_module.generate()
|
|
del generate_constants_module
|
|
|
|
# Read the version from a bare string stored in file in root directory
|
|
version = open(os.path.join(CProot,'.version'),'r').read().strip()
|
|
|
|
setup_kwargs = {}
|
|
from setuptools import setup, Extension, find_packages
|
|
if USE_CYTHON:
|
|
import Cython.Compiler
|
|
from Cython.Distutils.extension import Extension
|
|
from Cython.Build import cythonize
|
|
from Cython.Distutils import build_ext
|
|
|
|
# This will always generate HTML to show where there are still pythonic bits hiding out
|
|
Cython.Compiler.Options.annotate = True
|
|
|
|
setup_kwargs['cmdclass'] = dict(build_ext = build_ext)
|
|
|
|
print('Cython will be used; cy_ext is ' + cy_ext)
|
|
else:
|
|
print('Cython will not be used; cy_ext is ' + cy_ext)
|
|
|
|
def find_cpp_sources(root = os.path.join('..','..','src'), extensions = ['.cpp'], skip_files = None):
|
|
file_listing = []
|
|
for path, dirs, files in os.walk(root):
|
|
for file in files:
|
|
n,ext = os.path.splitext(file)
|
|
fname = os.path.relpath(os.path.join(path, file))
|
|
if skip_files is not None and fname in skip_files: continue
|
|
if ext in extensions:
|
|
file_listing.append(fname)
|
|
return file_listing
|
|
|
|
# Set variables for C++ sources and include directories
|
|
sources = find_cpp_sources(os.path.join(CProot,'src'), '*.cpp')
|
|
include_dirs = [
|
|
os.path.join(CProot),
|
|
os.path.join(CProot, 'include'),
|
|
os.path.join(CProot, 'src'),
|
|
os.path.join(CProot, 'externals', 'Eigen'),
|
|
os.path.join(CProot, 'externals', 'msgpack-c', 'include')]
|
|
|
|
## If the file is run directly without any parameters, clean, build and install
|
|
if len(sys.argv)==1:
|
|
sys.argv += ['clean', 'install']
|
|
|
|
common_args = dict(include_dirs = include_dirs,
|
|
language='c++')
|
|
|
|
if USE_CYTHON:
|
|
common_args.update(dict(cython_c_in_temp = True,
|
|
cython_directives = cython_directives
|
|
)
|
|
)
|
|
|
|
if STATIC_LIBRARY_BUILT == True:
|
|
CoolProp_module = Extension('CoolProp.CoolProp',
|
|
[os.path.join('CoolProp','CoolProp.' + cy_ext)],
|
|
libraries = ['CoolProp'],
|
|
library_dirs = [static_library_path],
|
|
**common_args)
|
|
else:
|
|
CoolProp_module = Extension('CoolProp.CoolProp',
|
|
[os.path.join('CoolProp','CoolProp.' + cy_ext)] + sources,
|
|
**common_args)
|
|
constants_module = Extension('CoolProp._constants',
|
|
[os.path.join('CoolProp','_constants.' + cy_ext)],
|
|
**common_args)
|
|
|
|
if not pypi:
|
|
copy_files()
|
|
|
|
ext_modules = [CoolProp_module, constants_module]
|
|
|
|
if USE_CYTHON:
|
|
ext_modules = cythonize(ext_modules, compiler_directives = cython_directives)
|
|
|
|
try:
|
|
setup (name = 'CoolProp',
|
|
version = version, # look above for the definition of version variable - don't modify it here
|
|
author = "Ian Bell",
|
|
author_email='ian.h.bell@gmail.com',
|
|
url='http://www.coolprop.org',
|
|
description = """Open-source thermodynamic and transport properties database""",
|
|
packages = find_packages(),
|
|
ext_modules = ext_modules,
|
|
package_dir = {'CoolProp':'CoolProp',},
|
|
package_data = {'CoolProp':['*.pxd',
|
|
'CoolPropBibTeXLibrary.bib',
|
|
'Plots/psyrc'] + recursive_collect_includes()},
|
|
classifiers = [
|
|
"Programming Language :: Python",
|
|
"Development Status :: 4 - Beta",
|
|
"Environment :: Other Environment",
|
|
"Intended Audience :: Developers",
|
|
"License :: OSI Approved :: MIT License",
|
|
"Operating System :: OS Independent",
|
|
"Topic :: Software Development :: Libraries :: Python Modules"
|
|
],
|
|
**setup_kwargs
|
|
)
|
|
except BaseException as E:
|
|
if not pypi:
|
|
remove_files()
|
|
raise
|
|
else:
|
|
if not pypi:
|
|
remove_files()
|