Add a "sticky" property to variants (#28630)
* Add sticky variants
* Add unit tests for sticky variants
* Add documentation for sticky variants
* Revert "Revert 19736 because conflicts are avoided by clingo by default (#26721)"
This reverts commit 33ef7d57c1.
* Add stickiness to "allow-unsupported-compiler"
			
			
This commit is contained in:
		
				
					committed by
					
						
						GitHub
					
				
			
			
				
	
			
			
			
						parent
						
							dd7acecf3d
						
					
				
				
					commit
					cd04109e17
				
			@@ -1441,6 +1441,32 @@ The ``when`` clause follows the same syntax and accepts the same
 | 
			
		||||
values as the ``when`` argument of
 | 
			
		||||
:py:func:`spack.directives.depends_on`
 | 
			
		||||
 | 
			
		||||
^^^^^^^^^^^^^^^
 | 
			
		||||
Sticky Variants
 | 
			
		||||
^^^^^^^^^^^^^^^
 | 
			
		||||
 | 
			
		||||
The variant directive can be marked as ``sticky`` by setting to ``True`` the
 | 
			
		||||
corresponding argument:
 | 
			
		||||
 | 
			
		||||
.. code-block:: python
 | 
			
		||||
 | 
			
		||||
   variant('bar', default=False, sticky=True)
 | 
			
		||||
 | 
			
		||||
A ``sticky`` variant differs from a regular one in that it is always set
 | 
			
		||||
to either:
 | 
			
		||||
 | 
			
		||||
#. An explicit value appearing in a spec literal or
 | 
			
		||||
#. Its default value
 | 
			
		||||
 | 
			
		||||
The concretizer thus is not free to pick an alternate value to work
 | 
			
		||||
around conflicts, but will error out instead.
 | 
			
		||||
Setting this property on a variant is useful in cases where the
 | 
			
		||||
variant allows some dangerous or controversial options (e.g. using unsupported versions
 | 
			
		||||
of a compiler for a library) and the packager wants to ensure that
 | 
			
		||||
allowing these options is done on purpose by the user, rather than
 | 
			
		||||
automatically by the solver.
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
^^^^^^^^^^^^^^^^^^^
 | 
			
		||||
Overriding Variants
 | 
			
		||||
^^^^^^^^^^^^^^^^^^^
 | 
			
		||||
 
 | 
			
		||||
@@ -5,6 +5,7 @@
 | 
			
		||||
 | 
			
		||||
import spack.variant
 | 
			
		||||
from spack.directives import conflicts, depends_on, variant
 | 
			
		||||
from spack.multimethod import when
 | 
			
		||||
from spack.package import PackageBase
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@@ -88,7 +89,7 @@ def cuda_flags(arch_list):
 | 
			
		||||
 | 
			
		||||
    # Linux x86_64 compiler conflicts from here:
 | 
			
		||||
    # https://gist.github.com/ax3l/9489132
 | 
			
		||||
 | 
			
		||||
    with when('^cuda~allow-unsupported-compilers'):
 | 
			
		||||
        # GCC
 | 
			
		||||
        # According to
 | 
			
		||||
        # https://github.com/spack/spack/pull/25054#issuecomment-886531664
 | 
			
		||||
 
 | 
			
		||||
@@ -562,7 +562,9 @@ def variant(
 | 
			
		||||
        values=None,
 | 
			
		||||
        multi=None,
 | 
			
		||||
        validator=None,
 | 
			
		||||
        when=None):
 | 
			
		||||
        when=None,
 | 
			
		||||
        sticky=False
 | 
			
		||||
):
 | 
			
		||||
    """Define a variant for the package. Packager can specify a default
 | 
			
		||||
    value as well as a text description.
 | 
			
		||||
 | 
			
		||||
@@ -583,7 +585,8 @@ def variant(
 | 
			
		||||
            doesn't meet the additional constraints
 | 
			
		||||
        when (spack.spec.Spec, bool): optional condition on which the
 | 
			
		||||
            variant applies
 | 
			
		||||
 | 
			
		||||
        sticky (bool): the variant should not be changed by the concretizer to
 | 
			
		||||
            find a valid concrete spec.
 | 
			
		||||
    Raises:
 | 
			
		||||
        DirectiveError: if arguments passed to the directive are invalid
 | 
			
		||||
    """
 | 
			
		||||
@@ -657,7 +660,7 @@ def _execute_variant(pkg):
 | 
			
		||||
            when_specs += orig_when
 | 
			
		||||
 | 
			
		||||
        pkg.variants[name] = (spack.variant.Variant(
 | 
			
		||||
            name, default, description, values, multi, validator
 | 
			
		||||
            name, default, description, values, multi, validator, sticky
 | 
			
		||||
        ), when_specs)
 | 
			
		||||
    return _execute_variant
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -852,6 +852,9 @@ def pkg_rules(self, pkg, tests):
 | 
			
		||||
            for value in sorted(values):
 | 
			
		||||
                self.gen.fact(fn.variant_possible_value(pkg.name, name, value))
 | 
			
		||||
 | 
			
		||||
            if variant.sticky:
 | 
			
		||||
                self.gen.fact(fn.variant_sticky(pkg.name, name))
 | 
			
		||||
 | 
			
		||||
            self.gen.newline()
 | 
			
		||||
 | 
			
		||||
        # conflicts
 | 
			
		||||
 
 | 
			
		||||
@@ -402,6 +402,14 @@ variant(Package, Variant) :- variant_condition(ID, Package, Variant),
 | 
			
		||||
   build(Package),
 | 
			
		||||
   error("Unsatisfied conditional variants cannot take on a variant value").
 | 
			
		||||
 | 
			
		||||
% if a variant is sticky and not set its value is the default value
 | 
			
		||||
variant_value(Package, Variant, Value) :-
 | 
			
		||||
  variant(Package, Variant),
 | 
			
		||||
  not variant_set(Package, Variant),
 | 
			
		||||
  variant_sticky(Package, Variant),
 | 
			
		||||
  variant_default_value(Package, Variant, Value),
 | 
			
		||||
  build(Package).
 | 
			
		||||
 | 
			
		||||
% one variant value for single-valued variants.
 | 
			
		||||
1 {
 | 
			
		||||
  variant_value(Package, Variant, Value)
 | 
			
		||||
@@ -523,6 +531,7 @@ variant_single_value(Package, "dev_path")
 | 
			
		||||
% spec or some package sets it, and without this, clingo will give
 | 
			
		||||
% warnings like 'info: atom does not occur in any rule head'.
 | 
			
		||||
#defined variant/2.
 | 
			
		||||
#defined variant_sticky/2.
 | 
			
		||||
#defined variant_set/3.
 | 
			
		||||
#defined variant_condition/3.
 | 
			
		||||
#defined variant_single_value/2.
 | 
			
		||||
 
 | 
			
		||||
@@ -1357,3 +1357,21 @@ def test_concrete_specs_are_not_modified_on_reuse(
 | 
			
		||||
        s = spack.spec.Spec(spec_str).concretized(reuse=True)
 | 
			
		||||
        assert s.package.installed is expect_installed
 | 
			
		||||
        assert s.satisfies(spec_str, strict=True)
 | 
			
		||||
 | 
			
		||||
    @pytest.mark.regression('26721,19736')
 | 
			
		||||
    def test_sticky_variant_in_package(self):
 | 
			
		||||
        if spack.config.get('config:concretizer') == 'original':
 | 
			
		||||
            pytest.skip('Original concretizer cannot use sticky variants')
 | 
			
		||||
 | 
			
		||||
        # Here we test that a sticky variant cannot be changed from its default value
 | 
			
		||||
        # by the ASP solver if not set explicitly. The package used in the test needs
 | 
			
		||||
        # to have +allow-gcc set to be concretized with %gcc and clingo is not allowed
 | 
			
		||||
        # to change the default ~allow-gcc
 | 
			
		||||
        with pytest.raises(spack.error.SpackError):
 | 
			
		||||
            spack.spec.Spec('sticky-variant %gcc').concretized()
 | 
			
		||||
 | 
			
		||||
        s = spack.spec.Spec('sticky-variant+allow-gcc %gcc').concretized()
 | 
			
		||||
        assert s.satisfies('%gcc') and s.satisfies('+allow-gcc')
 | 
			
		||||
 | 
			
		||||
        s = spack.spec.Spec('sticky-variant %clang').concretized()
 | 
			
		||||
        assert s.satisfies('%clang') and s.satisfies('~allow-gcc')
 | 
			
		||||
 
 | 
			
		||||
@@ -416,3 +416,13 @@ def test_multivalued_variants_are_lower_priority_than_providers(self):
 | 
			
		||||
        ):
 | 
			
		||||
            s = Spec('somevirtual').concretized()
 | 
			
		||||
            assert s.name == 'some-virtual-preferred'
 | 
			
		||||
 | 
			
		||||
    @pytest.mark.regression('26721,19736')
 | 
			
		||||
    def test_sticky_variant_accounts_for_packages_yaml(self):
 | 
			
		||||
        with spack.config.override(
 | 
			
		||||
                'packages:sticky-variant', {
 | 
			
		||||
                    'variants': '+allow-gcc'
 | 
			
		||||
                }
 | 
			
		||||
        ):
 | 
			
		||||
            s = Spec('sticky-variant %gcc').concretized()
 | 
			
		||||
            assert s.satisfies('%gcc') and s.satisfies('+allow-gcc')
 | 
			
		||||
 
 | 
			
		||||
@@ -37,7 +37,9 @@ def __init__(
 | 
			
		||||
            description,
 | 
			
		||||
            values=(True, False),
 | 
			
		||||
            multi=False,
 | 
			
		||||
            validator=None):
 | 
			
		||||
            validator=None,
 | 
			
		||||
            sticky=False
 | 
			
		||||
    ):
 | 
			
		||||
        """Initialize a package variant.
 | 
			
		||||
 | 
			
		||||
        Args:
 | 
			
		||||
@@ -51,6 +53,8 @@ def __init__(
 | 
			
		||||
            multi (bool): whether multiple CSV are allowed
 | 
			
		||||
            validator (callable): optional callable used to enforce
 | 
			
		||||
                additional logic on the set of values being validated
 | 
			
		||||
            sticky (bool): if true the variant is set to the default value at
 | 
			
		||||
                concretization time
 | 
			
		||||
        """
 | 
			
		||||
        self.name = name
 | 
			
		||||
        self.default = default
 | 
			
		||||
@@ -83,6 +87,7 @@ def isa_type(v):
 | 
			
		||||
 | 
			
		||||
        self.multi = multi
 | 
			
		||||
        self.group_validator = validator
 | 
			
		||||
        self.sticky = sticky
 | 
			
		||||
 | 
			
		||||
    def validate_or_raise(self, vspec, pkg=None):
 | 
			
		||||
        """Validate a variant spec against this package variant. Raises an
 | 
			
		||||
 
 | 
			
		||||
@@ -0,0 +1,18 @@
 | 
			
		||||
# Copyright 2013-2022 Lawrence Livermore National Security, LLC and other
 | 
			
		||||
# Spack Project Developers. See the top-level COPYRIGHT file for details.
 | 
			
		||||
#
 | 
			
		||||
# SPDX-License-Identifier: (Apache-2.0 OR MIT)
 | 
			
		||||
from spack import *
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class StickyVariant(AutotoolsPackage):
 | 
			
		||||
    """Package with a sticky variant and a conflict"""
 | 
			
		||||
 | 
			
		||||
    homepage = "http://www.example.com"
 | 
			
		||||
    url = "http://www.example.com/a-1.0.tar.gz"
 | 
			
		||||
 | 
			
		||||
    version('1.0', '0123456789abcdef0123456789abcdef')
 | 
			
		||||
 | 
			
		||||
    variant('allow-gcc', description='', default=False, sticky=True)
 | 
			
		||||
 | 
			
		||||
    conflicts('%gcc', when='~allow-gcc')
 | 
			
		||||
@@ -144,6 +144,8 @@ class Cuda(Package):
 | 
			
		||||
    conflicts('arch=darwin-mojave-x86_64')
 | 
			
		||||
 | 
			
		||||
    variant('dev', default=False, description='Enable development dependencies, i.e to use cuda-gdb')
 | 
			
		||||
    variant('allow-unsupported-compilers', default=False, sticky=True,
 | 
			
		||||
            description='Allow unsupported host compiler and CUDA version combinations')
 | 
			
		||||
 | 
			
		||||
    depends_on('libxml2', when='@10.1.243:')
 | 
			
		||||
    # cuda-gdb needed libncurses.so.5 before 11.4.0
 | 
			
		||||
 
 | 
			
		||||
		Reference in New Issue
	
	Block a user