Skip to content

Commit

Permalink
Cythonized files now deleted after setup. (closes #667)
Browse files Browse the repository at this point in the history
Default behavior now depends on release or dev status. When in dev mode
default is to use Cython and delete cythonized files. When in release
mode default is to skip Cython but keep existing cythonized files.

setup.py now reads the version string directly from version.py.

Environment variable handling in setup.py enhanced to recognize strings
with boolean meaning.
  • Loading branch information
mnmelo committed Feb 2, 2016
1 parent 34ce8fa commit 818e393
Show file tree
Hide file tree
Showing 3 changed files with 78 additions and 21 deletions.
11 changes: 9 additions & 2 deletions package/MDAnalysis/version.py
Original file line number Diff line number Diff line change
Expand Up @@ -58,5 +58,12 @@
# keep __version__ in separate file to avoid circular imports
# e.g. with lib.log

#: Release of MDAnalysis as a string, using `semantic versioning`_.
__version__ = "0.14.0-dev0" # NOTE: keep in sync with RELEASE in setup.py
# Release of MDAnalysis as a string, using `semantic versioning`_.

# In order for setup.py to know the current version this file will be
# exec'd during setup. Beware of adding any code other than the definition
# of the __version__ variable.

# setup.py will look for a 'dev' substring in __version__ to decide whether
# this is a release or development version. Please respect this nomenclature.
__version__ = "0.14.0-dev0"
8 changes: 6 additions & 2 deletions package/setup.cfg
Original file line number Diff line number Diff line change
@@ -1,6 +1,10 @@
[options]
#use_cython=False
#use_openmp=False
#debug_cflags=True
## The default values of use_cython and keep_cythonized
## depend on whether the current version is a release or not.
## Uncomment the following two lines to get the _release_ behavior.
#use_cython=False
#keep_cythonized=True
[wheel]
universal = 1
universal = 1
80 changes: 63 additions & 17 deletions package/setup.py
Original file line number Diff line number Diff line change
Expand Up @@ -67,6 +67,14 @@
cython_found = False
cmdclass = {}

# Find our own release code (by shamelessly running MDAnalysis/version.py,
# which brings the __version__ variable into the current namespace.)
with open("MDAnalysis/version.py") as f:
exec(f.read())
RELEASE = __version__

is_release = not 'dev' in RELEASE

if cython_found:
# cython has to be >=0.16 to support cython.parallel
import Cython
Expand All @@ -76,14 +84,16 @@
required_version = "0.16"

if not LooseVersion(Cython.__version__) >= LooseVersion(required_version):
raise ImportError(
"Cython version {0} (found {1}) is required because it offers "
"a handy parallelisation module".format(
required_version, Cython.__version__))
# We don't necessarily die here. Maybe we already have
# the cythonized '.c' files.
print("Cython version {0} was found but won't be used: version {1} "
"or greater is required because it offers a handy "
"parallelization module".format(
Cython.__version__, required_version))
cython_found = False
del Cython
del LooseVersion


class Config(object):
"""Config wrapper class to get build options
Expand All @@ -95,6 +105,9 @@ class Config(object):
3. given default
Environment variables should start with 'MDA_' and be all uppercase.
Values passed to environment variables are checked (case-insensitively)
for specific strings with boolean meaning: 'True' or '1' will cause `True`
to be returned. '0' or 'False' cause `False` to be returned.
"""

Expand All @@ -106,7 +119,12 @@ def __init__(self, fname='setup.cfg'):
def get(self, option_name, default=None):
environ_name = 'MDA_' + option_name.upper()
if environ_name in os.environ:
return os.environ[environ_name]
val = os.environ[environ_name]
if val.upper() in ('1', 'TRUE'):
return True
elif val.upper() in ('0', 'FALSE'):
return False
return val
try:
option = self.config.get('options', option_name)
return option
Expand Down Expand Up @@ -222,7 +240,8 @@ def detect_openmp():


def extensions(config):
use_cython = config.get('use_cython', default=True)
# dev installs must build their own cythonized files.
use_cython = config.get('use_cython', default=not is_release)
use_openmp = config.get('use_openmp', default=True)

if config.get('debug_cflags', default=False):
Expand Down Expand Up @@ -253,10 +272,13 @@ def extensions(config):
parallel_macros = [('PARALLEL', None)] if has_openmp and use_openmp else []

if use_cython:
print('Will attempt to use Cython.')
if not cython_found:
print("Couldn't find Cython installation. "
"Not recompiling cython extension")
print("Couldn't find a Cython installation. "
"Not recompiling cython extensions.")
use_cython = False
else:
print('Will not attempt to use Cython.')

source_suffix = '.pyx' if use_cython else '.c'

Expand Down Expand Up @@ -297,7 +319,7 @@ def extensions(config):
include_dirs=include_dirs,
extra_compile_args=extra_compile_args)
xdrlib = MDAExtension('lib.formats.xdrlib',
sources=['MDAnalysis/lib/formats/xdrlib.pyx',
sources=['MDAnalysis/lib/formats/xdrlib' + source_suffix,
'MDAnalysis/lib/formats/src/xdrfile.c',
'MDAnalysis/lib/formats/src/xdrfile_xtc.c',
'MDAnalysis/lib/formats/src/xdrfile_trr.c',
Expand All @@ -308,18 +330,31 @@ def extensions(config):
'MDAnalysis/lib/formats'],
define_macros=largefile_macros)
util = MDAExtension('lib.formats.cython_util',
sources=['MDAnalysis/lib/formats/cython_util.pyx'],
sources=['MDAnalysis/lib/formats/cython_util' + source_suffix],
include_dirs=include_dirs)

extensions = [dcd, dcd_time, distances, distances_omp, qcprot,
pre_exts = [dcd, dcd_time, distances, distances_omp, qcprot,
transformation, xdrlib, util]
cython_generated = []
if use_cython:
extensions = cythonize(extensions)
return extensions
extensions = cythonize(pre_exts)
for pre_ext, post_ext in zip(pre_exts, extensions):
for source in post_ext.sources:
if source not in pre_ext.sources:
cython_generated.append(source)
else:
#Let's check early for missing .c files
extensions = pre_exts
for ext in extensions:
for source in ext.sources:
if not (os.path.isfile(source) and
os.access(source, os.R_OK)):
raise IOError("Source file '{}' not found. This might be "
"caused by a missing Cython install, or a "
"failed/disabled Cython build.".format(source))
return extensions, cython_generated

if __name__ == '__main__':
# NOTE: keep in sync with MDAnalysis.__version__ in version.py
RELEASE = "0.14.0-dev0"
with open("SUMMARY.txt") as summary:
LONG_DESCRIPTION = summary.read()
CLASSIFIERS = [
Expand All @@ -337,6 +372,7 @@ def extensions(config):
]

config = Config()
exts, cythonfiles = extensions(config)

setup(name='MDAnalysis',
version=RELEASE,
Expand All @@ -353,7 +389,7 @@ def extensions(config):
packages=find_packages(),
package_dir={'MDAnalysis': 'MDAnalysis'},
ext_package='MDAnalysis',
ext_modules=extensions(config),
ext_modules=exts,
classifiers=CLASSIFIERS,
cmdclass=cmdclass,
requires=['numpy (>=1.5.0)', 'biopython',
Expand Down Expand Up @@ -391,3 +427,13 @@ def extensions(config):
zip_safe=False, # as a zipped egg the *.so files are not found (at
# least in Ubuntu/Linux)
)

# Releases keep their cythonized stuff for shipping.
if not config.get('keep_cythonized', default=is_release):
for cythonized in cythonfiles:
try:
os.unlink(cythonized)
except OSError as err:
print("Warning: failed to delete cythonized file {}: {}. "
"Moving on.".format(cythonized, err.strerror))

0 comments on commit 818e393

Please sign in to comment.