#!/usr/bin/env python
# encoding: utf-8
# Andre Anjos <andre.anjos@idiap.ch>
# Thu Mar 20 12:38:14 CET 2014
"""Helps looking for Boost on stock file-system locations"""
import os
import re
import sys
import glob
from distutils.version import LooseVersion
from .utils import uniq, egrep, find_header, find_library
def boost_version(version_hpp):
matches = egrep(version_hpp, r"^#\s*define\s+BOOST_VERSION\s+(\d+)\s*$")
if not len(matches): return None
# we have a match, produce a string version of the version number
version_int = int(matches[0].group(1))
version_tuple = (
version_int // 100000,
(version_int // 100) % 1000,
version_int % 100,
)
return '.'.join([str(k) for k in version_tuple])
[docs]class boost:
"""A class for capturing configuration information from boost
Example usage:
.. doctest::
:options: +NORMALIZE_WHITESPACE +ELLIPSIS
>>> from bob.extension import boost
>>> pkg = boost('>= 1.35')
>>> pkg.include_directory
'...'
>>> pkg.version
'...'
You can also use this class to retrieve information about installed Boost
libraries and link information:
.. doctest::
:options: +NORMALIZE_WHITESPACE +ELLIPSIS
>>> from bob.extension import boost
>>> pkg = boost('>= 1.35')
>>> pkg.libconfig(['python', 'system'])
(...)
"""
def __init__ (self, requirement=''):
"""
Searches for the Boost library in stock locations. Allows user to override.
If the user sets the environment variable BOB_PREFIX_PATH, that prefixes
the standard path locations.
"""
candidates = find_header('version.hpp', subpaths=['boost', 'boost?*'])
if not candidates:
raise RuntimeError("could not find boost's `version.hpp' - have you installed Boost on this machine?")
found = False
if not requirement:
# since we use boost headers **including the boost/ directory**, we need to go one level lower
self.include_directory = os.path.dirname(os.path.dirname(candidates[0]))
self.version = boost_version(candidates[0])
found = True
else:
# requirement is 'operator' 'version'
operator, required = [k.strip() for k in requirement.split(' ', 1)]
# now check for user requirements
for path in candidates:
version = boost_version(path)
available = LooseVersion(version)
if (operator == '<' and available < required) or \
(operator == '<=' and available <= required) or \
(operator == '>' and available > required) or \
(operator == '>=' and available >= required) or \
(operator == '==' and available == required):
self.include_directory = path
self.version = version
found = True
break
if not found:
raise RuntimeError("could not find the required (%s) version of boost on the file system (looked at: %s)" % (requirement, ', '.join(candidates)))
# normalize
self.include_directory = os.path.normpath(self.include_directory)
[docs] def libconfig(self, modules, only_static=False,
templates=['boost_%(name)s-mt-%(py)s', 'boost_%(name)s-%(py)s', 'boost_%(name)s-mt', 'boost_%(name)s']):
"""Returns a tuple containing the library configuration for requested
modules.
This function respects the path location where the include files for Boost
are installed.
Parameters:
modules (list of strings)
A list of string specifying the requested libraries to search for. For
example, to search for `libboost_mpi.so`, pass only ``mpi``.
static (bool)
A boolean, indicating if we should try only to search for static versions
of the libraries. If not set, any would do.
templates (list of template strings)
A list that defines in which order to search for libraries on the default
search path, defined by ``self.include_directory``. Tune this list if you
have compiled specific versions of Boost with support to multi-threading
(``-mt``), debug (``-g``), STLPORT (``-p``) or required to insert
compiler, the underlying thread API used or your own namespace.
Here are the keywords you can use:
%(name)s
resolves to the module name you are searching for
%(ver)s
resolves to the current boost version string (e.g. ``'1.50.0'``)
%(py)s
resolves to the string ``'pyXY'`` where ``XY`` represent the major and
minor versions of the current python interpreter.
Example templates:
* ``'boost_%(name)s-mt'``
* ``'boost_%(name)s'``
* ``'boost_%(name)s-gcc43-%(ver)s'``
Returns:
directories (list of strings)
A list of directories indicating where the libraries are installed
libs (list of strings)
A list of strings indicating the names of the libraries you can use
"""
# make the include header prefix preferential
prefix = os.path.dirname(self.include_directory)
py = 'py%d%d' % sys.version_info[:2]
filenames = []
for module in modules:
candidates = []
modnames = [k % dict(name=module, ver=self.version, py=py) for k in
templates]
for modname in modnames:
candidates += find_library(modname, version=self.version,
prefixes=[prefix], only_static=only_static)
if not candidates:
raise RuntimeError("cannot find required boost module `%s' - make sure boost is installed on `%s' and that this module is named %s on the filesystem" % (module, prefix, ' or '.join(modnames)))
# take the first choice that includes the prefix (or the absolute first choice otherwise)
index = 0
for i, candidate in enumerate(candidates):
if candidate.find(prefix) == 0:
index = i
break
filenames.append(candidates[index])
# libraries
libraries = []
for f in filenames:
name, ext = os.path.splitext(os.path.basename(f))
if ext in ['.so', '.a', '.dylib', '.dll']:
libraries.append(name[3:]) #strip 'lib' from the name
else: #link against the whole thing
libraries.append(':' + os.path.basename(f))
# library paths
libpaths = [os.path.dirname(k) for k in filenames]
return uniq(libpaths), uniq(libraries)
[docs] def macros(self):
"""Returns package availability and version number macros
This method returns a python list with 2 macros indicating package
availability and a version number, using standard GNU compatible names.
Example:
.. doctest::
:options: +NORMALIZE_WHITESPACE +ELLIPSIS
>>> from bob.extension import boost
>>> pkg = boost('>= 1.34')
>>> pkg.macros()
[('HAVE_BOOST', '1')]
"""
return [('HAVE_BOOST', '1')]