
* Transferred pending changes from efischer/develop * 1. Rewrite of "Getting Started": everything you need to set up Spack, even on old/ornery systems. This is not a reference manual section; items covered here are covered more systematically elsewhere in the manual. Some sections were moved here from elsewhere. 2. Beginning to write three methods of application developer support. Two methods were moved from elsewhere. * Edits... * Moved sections in preparation for additional text to be added from old efischer/docs branch. * Moved 2 more sections. * Avoid accid * Applied proofreading edits from @adamjstewart * Fixed non-standard section characters. * Moved section on profiling to the developer's guide. * Still working on Spack workflows... * Finished draft of packaging_guide.rst * Renamed sample projects. * Updates to docstrings * Added documentation to resolve #638 (content taken from #846) * Added section on resolving inconsistent run dependencies. Addresses #645 * Showed how to build Python extensions only compatible with certain versions of Python. * Added examples of getting the right behavior from depends_on(). See #1035 * Added section on Intel compilers and their GCC masquerading feature. Addresses #638, #1687. * Fixed formatting * Added fixes to filesystem views. Added a caveats section to ``spack setup``. * Updated section on Intel compiler configuration because compiler flags currently do not work (see #1687) * Defined trusted downloads, and updated text based on them. (See #1696) * Added workflow to deal with buggy upstream software. See #1683 * Added proper separation between Spack Docs vs. Reference Manual * Renamed spack_workflows to workflows. Resolves a conflict with the .gitignore file. * Removed repeated section. * Created new "Vendor Specific Compiler Configuration" section and organized existing Intel section into it. Added new PGI and NAG sections; but they need to be expanded / rewritten based on the existing text plus research through Spack issues on GitHub. * Fixed text on `spack load --dependencies` to conform to reality. See #1662 * Added patching as option for upstream bugfixes. * Added section on using licensed compilers. * Added section on non-downloadable tarballs. * Wrote sections on NAG and PGI. Arranged compilers in alphabetical order. * Fix indent. * Fixed typos. * Clarified dependency types. * Applied edits from Adam J. Stewart. Spellchecked workflows and getting_started. * Removed spurious header * Fixed Sphinx errors * Fixed erroneous symbol in docstring. * Fix many typos and formatting problems. * Spacing changes * Added section on fixing Git problems. See #1779 * Fixed signature of install() method. * Addressed system packages in greater detail. See #1794 #1795 * Fixed typos * Fixed quotes * Duplicate section on Spack profiling removed from configuration.rst. It had earlier been moved to developer_guide.rst, where it fits better. * Minor edits - Tweak supported platform language. - Various small changes to the new getting started guide. * Fixed bug with quotes.
361 lines
13 KiB
Python
361 lines
13 KiB
Python
##############################################################################
|
|
# Copyright (c) 2013-2016, Lawrence Livermore National Security, LLC.
|
|
# Produced at the Lawrence Livermore National Laboratory.
|
|
#
|
|
# This file is part of Spack.
|
|
# Created by Todd Gamblin, tgamblin@llnl.gov, All rights reserved.
|
|
# LLNL-CODE-647188
|
|
#
|
|
# For details, see https://github.com/llnl/spack
|
|
# Please also see the LICENSE file for our notice and the LGPL.
|
|
#
|
|
# This program is free software; you can redistribute it and/or modify
|
|
# it under the terms of the GNU Lesser General Public License (as
|
|
# published by the Free Software Foundation) version 2.1, February 1999.
|
|
#
|
|
# This program is distributed in the hope that it will be useful, but
|
|
# WITHOUT ANY WARRANTY; without even the IMPLIED WARRANTY OF
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the terms and
|
|
# conditions of the GNU Lesser General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU Lesser General Public
|
|
# License along with this program; if not, write to the Free Software
|
|
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
##############################################################################
|
|
"""This package contains directives that can be used within a package.
|
|
|
|
Directives are functions that can be called inside a package
|
|
definition to modify the package, for example:
|
|
|
|
class OpenMpi(Package):
|
|
depends_on("hwloc")
|
|
provides("mpi")
|
|
...
|
|
|
|
``provides`` and ``depends_on`` are spack directives.
|
|
|
|
The available directives are:
|
|
|
|
* ``version``
|
|
* ``depends_on``
|
|
* ``provides``
|
|
* ``extends``
|
|
* ``patch``
|
|
* ``variant``
|
|
* ``resource``
|
|
|
|
"""
|
|
|
|
import re
|
|
import os.path
|
|
import functools
|
|
|
|
from llnl.util.lang import *
|
|
from llnl.util.filesystem import join_path
|
|
|
|
import spack
|
|
import spack.spec
|
|
import spack.error
|
|
import spack.url
|
|
from spack.version import Version
|
|
from spack.patch import Patch
|
|
from spack.variant import Variant
|
|
from spack.spec import Spec, parse_anonymous_spec
|
|
from spack.resource import Resource
|
|
from spack.fetch_strategy import from_kwargs
|
|
|
|
__all__ = ['depends_on', 'extends', 'provides', 'patch', 'version', 'variant',
|
|
'resource']
|
|
|
|
#
|
|
# This is a list of all directives, built up as they are defined in
|
|
# this file.
|
|
#
|
|
directives = {}
|
|
|
|
|
|
def ensure_dicts(pkg):
|
|
"""Ensure that a package has all the dicts required by directives."""
|
|
for name, d in directives.items():
|
|
d.ensure_dicts(pkg)
|
|
|
|
|
|
class directive(object):
|
|
"""Decorator for Spack directives.
|
|
|
|
Spack directives allow you to modify a package while it is being
|
|
defined, e.g. to add version or dependency information. Directives
|
|
are one of the key pieces of Spack's package "language", which is
|
|
embedded in python.
|
|
|
|
Here's an example directive:
|
|
|
|
@directive(dicts='versions')
|
|
version(pkg, ...):
|
|
...
|
|
|
|
This directive allows you write:
|
|
|
|
class Foo(Package):
|
|
version(...)
|
|
|
|
The ``@directive`` decorator handles a couple things for you:
|
|
|
|
1. Adds the class scope (pkg) as an initial parameter when
|
|
called, like a class method would. This allows you to modify
|
|
a package from within a directive, while the package is still
|
|
being defined.
|
|
|
|
2. It automatically adds a dictionary called "versions" to the
|
|
package so that you can refer to pkg.versions.
|
|
|
|
The ``(dicts='versions')`` part ensures that ALL packages in Spack
|
|
will have a ``versions`` attribute after they're constructed, and
|
|
that if no directive actually modified it, it will just be an
|
|
empty dict.
|
|
|
|
This is just a modular way to add storage attributes to the
|
|
Package class, and it's how Spack gets information from the
|
|
packages to the core.
|
|
|
|
"""
|
|
|
|
def __init__(self, dicts=None):
|
|
if isinstance(dicts, basestring):
|
|
dicts = (dicts, )
|
|
elif type(dicts) not in (list, tuple):
|
|
raise TypeError(
|
|
"dicts arg must be list, tuple, or string. Found %s" %
|
|
type(dicts))
|
|
|
|
self.dicts = dicts
|
|
|
|
def ensure_dicts(self, pkg):
|
|
"""Ensure that a package has the dicts required by this directive."""
|
|
for d in self.dicts:
|
|
if not hasattr(pkg, d):
|
|
setattr(pkg, d, {})
|
|
|
|
attr = getattr(pkg, d)
|
|
if not isinstance(attr, dict):
|
|
raise spack.error.SpackError(
|
|
"Package %s has non-dict %s attribute!" % (pkg, d))
|
|
|
|
def __call__(self, directive_function):
|
|
directives[directive_function.__name__] = self
|
|
|
|
@functools.wraps(directive_function)
|
|
def wrapped(*args, **kwargs):
|
|
pkg = DictWrapper(caller_locals())
|
|
self.ensure_dicts(pkg)
|
|
|
|
pkg.name = get_calling_module_name()
|
|
return directive_function(pkg, *args, **kwargs)
|
|
|
|
return wrapped
|
|
|
|
|
|
@directive('versions')
|
|
def version(pkg, ver, checksum=None, **kwargs):
|
|
"""Adds a version and metadata describing how to fetch it.
|
|
Metadata is just stored as a dict in the package's versions
|
|
dictionary. Package must turn it into a valid fetch strategy
|
|
later.
|
|
"""
|
|
# TODO: checksum vs md5 distinction is confusing -- fix this.
|
|
# special case checksum for backward compatibility
|
|
if checksum:
|
|
kwargs['md5'] = checksum
|
|
|
|
# Store kwargs for the package to later with a fetch_strategy.
|
|
pkg.versions[Version(ver)] = kwargs
|
|
|
|
|
|
def _depends_on(pkg, spec, when=None, type=None):
|
|
# If when is False do nothing
|
|
if when is False:
|
|
return
|
|
# If when is None or True make sure the condition is always satisfied
|
|
if when is None or when is True:
|
|
when = pkg.name
|
|
when_spec = parse_anonymous_spec(when, pkg.name)
|
|
|
|
if type is None:
|
|
# The default deptype is build and link because the common case is to
|
|
# build against a library which then turns into a runtime dependency
|
|
# due to the linker.
|
|
# XXX(deptype): Add 'run' to this? It's an uncommon dependency type,
|
|
# but is most backwards-compatible.
|
|
type = ('build', 'link')
|
|
|
|
if isinstance(type, str):
|
|
type = spack.spec.special_types.get(type, (type,))
|
|
|
|
for deptype in type:
|
|
if deptype not in spack.spec.alldeps:
|
|
raise UnknownDependencyTypeError('depends_on', pkg.name, deptype)
|
|
|
|
dep_spec = Spec(spec)
|
|
if pkg.name == dep_spec.name:
|
|
raise CircularReferenceError('depends_on', pkg.name)
|
|
|
|
pkg_deptypes = pkg._deptypes.setdefault(dep_spec.name, set())
|
|
for deptype in type:
|
|
pkg_deptypes.add(deptype)
|
|
|
|
conditions = pkg.dependencies.setdefault(dep_spec.name, {})
|
|
if when_spec in conditions:
|
|
conditions[when_spec].constrain(dep_spec, deps=False)
|
|
else:
|
|
conditions[when_spec] = dep_spec
|
|
|
|
|
|
@directive(('dependencies', '_deptypes'))
|
|
def depends_on(pkg, spec, when=None, type=None):
|
|
"""Creates a dict of deps with specs defining when they apply.
|
|
This directive is to be used inside a Package definition to declare
|
|
that the package requires other packages to be built first.
|
|
@see The section "Dependency specs" in the Spack Packaging Guide."""
|
|
_depends_on(pkg, spec, when=when, type=type)
|
|
|
|
|
|
@directive(('extendees', 'dependencies', '_deptypes'))
|
|
def extends(pkg, spec, **kwargs):
|
|
"""Same as depends_on, but dependency is symlinked into parent prefix.
|
|
|
|
This is for Python and other language modules where the module
|
|
needs to be installed into the prefix of the Python installation.
|
|
Spack handles this by installing modules into their own prefix,
|
|
but allowing ONE module version to be symlinked into a parent
|
|
Python install at a time.
|
|
|
|
keyword arguments can be passed to extends() so that extension
|
|
packages can pass parameters to the extendee's extension
|
|
mechanism.
|
|
|
|
"""
|
|
if pkg.extendees:
|
|
raise DirectiveError("Packages can extend at most one other package.")
|
|
|
|
when = kwargs.pop('when', pkg.name)
|
|
_depends_on(pkg, spec, when=when)
|
|
pkg.extendees[spec] = (Spec(spec), kwargs)
|
|
|
|
|
|
@directive('provided')
|
|
def provides(pkg, *specs, **kwargs):
|
|
"""Allows packages to provide a virtual dependency. If a package provides
|
|
'mpi', other packages can declare that they depend on "mpi", and spack
|
|
can use the providing package to satisfy the dependency.
|
|
"""
|
|
spec_string = kwargs.get('when', pkg.name)
|
|
provider_spec = parse_anonymous_spec(spec_string, pkg.name)
|
|
|
|
for string in specs:
|
|
for provided_spec in spack.spec.parse(string):
|
|
if pkg.name == provided_spec.name:
|
|
raise CircularReferenceError('depends_on', pkg.name)
|
|
pkg.provided[provided_spec] = provider_spec
|
|
|
|
|
|
@directive('patches')
|
|
def patch(pkg, url_or_filename, level=1, when=None):
|
|
"""Packages can declare patches to apply to source. You can
|
|
optionally provide a when spec to indicate that a particular
|
|
patch should only be applied when the package's spec meets
|
|
certain conditions (e.g. a particular version).
|
|
"""
|
|
if when is None:
|
|
when = pkg.name
|
|
when_spec = parse_anonymous_spec(when, pkg.name)
|
|
cur_patches = pkg.patches.setdefault(when_spec, [])
|
|
# if this spec is identical to some other, then append this
|
|
# patch to the existing list.
|
|
cur_patches.append(Patch(pkg, url_or_filename, level))
|
|
|
|
|
|
@directive('variants')
|
|
def variant(pkg, name, default=False, description=""):
|
|
"""Define a variant for the package. Packager can specify a default
|
|
value (on or off) as well as a text description."""
|
|
|
|
default = default
|
|
description = str(description).strip()
|
|
|
|
if not re.match(spack.spec.identifier_re, name):
|
|
raise DirectiveError("Invalid variant name in %s: '%s'" %
|
|
(pkg.name, name))
|
|
|
|
pkg.variants[name] = Variant(default, description)
|
|
|
|
|
|
@directive('resources')
|
|
def resource(pkg, **kwargs):
|
|
"""Define an external resource to be fetched and staged when building the
|
|
package. Based on the keywords present in the dictionary the appropriate
|
|
FetchStrategy will be used for the resource. Resources are fetched and
|
|
staged in their own folder inside spack stage area, and then moved into
|
|
the stage area of the package that needs them.
|
|
|
|
List of recognized keywords:
|
|
|
|
* 'when' : (optional) represents the condition upon which the resource is
|
|
needed
|
|
* 'destination' : (optional) path where to move the resource. This path
|
|
must be relative to the main package stage area.
|
|
* 'placement' : (optional) gives the possibility to fine tune how the
|
|
resource is moved into the main package stage area.
|
|
"""
|
|
when = kwargs.get('when', pkg.name)
|
|
destination = kwargs.get('destination', "")
|
|
placement = kwargs.get('placement', None)
|
|
# Check if the path is relative
|
|
if os.path.isabs(destination):
|
|
message = "The destination keyword of a resource directive can't be"
|
|
" an absolute path.\n"
|
|
message += "\tdestination : '{dest}\n'".format(dest=destination)
|
|
raise RuntimeError(message)
|
|
# Check if the path falls within the main package stage area
|
|
test_path = 'stage_folder_root'
|
|
normalized_destination = os.path.normpath(join_path(test_path, destination)
|
|
) # Normalized absolute path
|
|
if test_path not in normalized_destination:
|
|
message = "The destination folder of a resource must fall within the"
|
|
" main package stage directory.\n"
|
|
message += "\tdestination : '{dest}'\n".format(dest=destination)
|
|
raise RuntimeError(message)
|
|
when_spec = parse_anonymous_spec(when, pkg.name)
|
|
resources = pkg.resources.setdefault(when_spec, [])
|
|
name = kwargs.get('name')
|
|
fetcher = from_kwargs(**kwargs)
|
|
resources.append(Resource(name, fetcher, destination, placement))
|
|
|
|
|
|
class DirectiveError(spack.error.SpackError):
|
|
"""This is raised when something is wrong with a package directive."""
|
|
|
|
def __init__(self, directive, message):
|
|
super(DirectiveError, self).__init__(message)
|
|
self.directive = directive
|
|
|
|
|
|
class CircularReferenceError(DirectiveError):
|
|
"""This is raised when something depends on itself."""
|
|
|
|
def __init__(self, directive, package):
|
|
super(CircularReferenceError, self).__init__(
|
|
directive,
|
|
"Package '%s' cannot pass itself to %s" % (package, directive))
|
|
self.package = package
|
|
|
|
|
|
class UnknownDependencyTypeError(DirectiveError):
|
|
"""This is raised when a dependency is of an unknown type."""
|
|
|
|
def __init__(self, directive, package, deptype):
|
|
super(UnknownDependencyTypeError, self).__init__(
|
|
directive,
|
|
"Package '%s' cannot depend on a package via %s."
|
|
% (package, deptype))
|
|
self.package = package
|