diff options
author | Massimiliano Culpo <massimiliano.culpo@gmail.com> | 2024-11-05 07:46:49 +0100 |
---|---|---|
committer | GitHub <noreply@github.com> | 2024-11-05 07:46:49 +0100 |
commit | 14bc900e9db7ddaaf32d79c32386a9c81f65c18b (patch) | |
tree | 371b07737aae730c7e5eb677bf3a4d02299c0bd0 /lib | |
parent | e42e54160516d59a65ca9dc4c94cae9182aada41 (diff) | |
download | spack-14bc900e9db7ddaaf32d79c32386a9c81f65c18b.tar.gz spack-14bc900e9db7ddaaf32d79c32386a9c81f65c18b.tar.bz2 spack-14bc900e9db7ddaaf32d79c32386a9c81f65c18b.tar.xz spack-14bc900e9db7ddaaf32d79c32386a9c81f65c18b.zip |
spack.concretize: add type-hints, remove kwargs (#47382)
Also remove find_spec, which was used by the old concretizer.
Currently, it seems to be used only in tests.
Diffstat (limited to 'lib')
-rw-r--r-- | lib/spack/docs/conf.py | 1 | ||||
-rw-r--r-- | lib/spack/spack/cmd/__init__.py | 2 | ||||
-rw-r--r-- | lib/spack/spack/concretize.py | 117 | ||||
-rw-r--r-- | lib/spack/spack/environment/environment.py | 18 | ||||
-rw-r--r-- | lib/spack/spack/test/concretize.py | 38 |
5 files changed, 52 insertions, 124 deletions
diff --git a/lib/spack/docs/conf.py b/lib/spack/docs/conf.py index 4873e3e104..18495d4bca 100644 --- a/lib/spack/docs/conf.py +++ b/lib/spack/docs/conf.py @@ -214,6 +214,7 @@ nitpick_ignore = [ # Spack classes that intersphinx is unable to resolve ("py:class", "spack.version.StandardVersion"), ("py:class", "spack.spec.DependencySpec"), + ("py:class", "spack.spec.ArchSpec"), ("py:class", "spack.spec.InstallStatus"), ("py:class", "spack.spec.SpecfileReaderBase"), ("py:class", "spack.install_test.Pb"), diff --git a/lib/spack/spack/cmd/__init__.py b/lib/spack/spack/cmd/__init__.py index 031b29f952..7cf032c907 100644 --- a/lib/spack/spack/cmd/__init__.py +++ b/lib/spack/spack/cmd/__init__.py @@ -194,7 +194,7 @@ def _concretize_spec_pairs(to_concretize, tests=False): elif unify == "when_possible": concretize_method = spack.concretize.concretize_together_when_possible - concretized = concretize_method(*to_concretize, tests=tests) + concretized = concretize_method(to_concretize, tests=tests) return [concrete for _, concrete in concretized] diff --git a/lib/spack/spack/concretize.py b/lib/spack/spack/concretize.py index fabfdbb523..122e6c59c0 100644 --- a/lib/spack/spack/concretize.py +++ b/lib/spack/spack/concretize.py @@ -2,20 +2,17 @@ # Spack Project Developers. See the top-level COPYRIGHT file for details. # # SPDX-License-Identifier: (Apache-2.0 OR MIT) -""" -(DEPRECATED) Used to contain the code for the original concretizer -""" +"""High-level functions to concretize list of specs""" import sys import time from contextlib import contextmanager -from itertools import chain -from typing import Tuple +from typing import Iterable, Optional, Sequence, Tuple, Union import llnl.util.tty as tty import spack.config import spack.error -from spack.spec import Spec +from spack.spec import ArchSpec, CompilerSpec, Spec CHECK_COMPILER_EXISTENCE = True @@ -36,91 +33,59 @@ def enable_compiler_existence_check(): CHECK_COMPILER_EXISTENCE = saved -def find_spec(spec, condition, default=None): - """Searches the dag from spec in an intelligent order and looks - for a spec that matches a condition""" - # First search parents, then search children - deptype = ("build", "link") - dagiter = chain( - spec.traverse(direction="parents", deptype=deptype, root=False), - spec.traverse(direction="children", deptype=deptype, root=False), - ) - visited = set() - for relative in dagiter: - if condition(relative): - return relative - visited.add(id(relative)) - - # Then search all other relatives in the DAG *except* spec - for relative in spec.root.traverse(deptype="all"): - if relative is spec: - continue - if id(relative) in visited: - continue - if condition(relative): - return relative - - # Finally search spec itself. - if condition(spec): - return spec - - return default # Nothing matched the condition; return default. - - -def concretize_specs_together(*abstract_specs, **kwargs): +SpecPair = Tuple[Spec, Spec] +SpecLike = Union[Spec, str] +TestsType = Union[bool, Iterable[str]] + + +def concretize_specs_together( + abstract_specs: Sequence[SpecLike], tests: TestsType = False +) -> Sequence[Spec]: """Given a number of specs as input, tries to concretize them together. Args: - tests (bool or list or set): False to run no tests, True to test - all packages, or a list of package names to run tests for some - *abstract_specs: abstract specs to be concretized, given either - as Specs or strings - - Returns: - List of concretized specs + abstract_specs: abstract specs to be concretized + tests: list of package names for which to consider tests dependencies. If True, all nodes + will have test dependencies. If False, test dependencies will be disregarded. """ import spack.solver.asp allow_deprecated = spack.config.get("config:deprecated", False) solver = spack.solver.asp.Solver() - result = solver.solve( - abstract_specs, tests=kwargs.get("tests", False), allow_deprecated=allow_deprecated - ) + result = solver.solve(abstract_specs, tests=tests, allow_deprecated=allow_deprecated) return [s.copy() for s in result.specs] -def concretize_together(*spec_list, **kwargs): +def concretize_together( + spec_list: Sequence[SpecPair], tests: TestsType = False +) -> Sequence[SpecPair]: """Given a number of specs as input, tries to concretize them together. Args: - tests (bool or list or set): False to run no tests, True to test - all packages, or a list of package names to run tests for some - *spec_list: list of tuples to concretize. First entry is abstract spec, second entry is + spec_list: list of tuples to concretize. First entry is abstract spec, second entry is already concrete spec or None if not yet concretized - - Returns: - List of tuples of abstract and concretized specs + tests: list of package names for which to consider tests dependencies. If True, all nodes + will have test dependencies. If False, test dependencies will be disregarded. """ to_concretize = [concrete if concrete else abstract for abstract, concrete in spec_list] abstract_specs = [abstract for abstract, _ in spec_list] - concrete_specs = concretize_specs_together(*to_concretize, **kwargs) + concrete_specs = concretize_specs_together(to_concretize, tests=tests) return list(zip(abstract_specs, concrete_specs)) -def concretize_together_when_possible(*spec_list, **kwargs): +def concretize_together_when_possible( + spec_list: Sequence[SpecPair], tests: TestsType = False +) -> Sequence[SpecPair]: """Given a number of specs as input, tries to concretize them together to the extent possible. See documentation for ``unify: when_possible`` concretization for the precise definition of "to the extent possible". Args: - tests (bool or list or set): False to run no tests, True to test - all packages, or a list of package names to run tests for some - *spec_list: list of tuples to concretize. First entry is abstract spec, second entry is + spec_list: list of tuples to concretize. First entry is abstract spec, second entry is already concrete spec or None if not yet concretized - - Returns: - List of tuples of abstract and concretized specs + tests: list of package names for which to consider tests dependencies. If True, all nodes + will have test dependencies. If False, test dependencies will be disregarded. """ to_concretize = [concrete if concrete else abstract for abstract, concrete in spec_list] old_concrete_to_abstract = { @@ -131,7 +96,7 @@ def concretize_together_when_possible(*spec_list, **kwargs): solver = spack.solver.asp.Solver() allow_deprecated = spack.config.get("config:deprecated", False) for result in solver.solve_in_rounds( - to_concretize, tests=kwargs.get("tests", False), allow_deprecated=allow_deprecated + to_concretize, tests=tests, allow_deprecated=allow_deprecated ): result_by_user_spec.update(result.specs_by_input) @@ -143,19 +108,17 @@ def concretize_together_when_possible(*spec_list, **kwargs): ] -def concretize_separately(*spec_list, **kwargs): - """Given a number of specs as input, tries to concretize them together. +def concretize_separately( + spec_list: Sequence[SpecPair], tests: TestsType = False +) -> Sequence[SpecPair]: + """Concretizes the input specs separately from each other. Args: - tests (bool or list or set): False to run no tests, True to test - all packages, or a list of package names to run tests for some - *spec_list: list of tuples to concretize. First entry is abstract spec, second entry is + spec_list: list of tuples to concretize. First entry is abstract spec, second entry is already concrete spec or None if not yet concretized - - Returns: - List of tuples of abstract and concretized specs + tests: list of package names for which to consider tests dependencies. If True, all nodes + will have test dependencies. If False, test dependencies will be disregarded. """ - tests = kwargs.get("tests", False) to_concretize = [abstract for abstract, concrete in spec_list if not concrete] args = [ (i, str(abstract), tests) @@ -215,7 +178,7 @@ def concretize_separately(*spec_list, **kwargs): ] -def _concretize_task(packed_arguments) -> Tuple[int, Spec, float]: +def _concretize_task(packed_arguments: Tuple[int, str, TestsType]) -> Tuple[int, Spec, float]: index, spec_str, tests = packed_arguments with tty.SuppressOutput(msg_enabled=False): start = time.time() @@ -227,10 +190,10 @@ class UnavailableCompilerVersionError(spack.error.SpackError): """Raised when there is no available compiler that satisfies a compiler spec.""" - def __init__(self, compiler_spec, arch=None): - err_msg = "No compilers with spec {0} found".format(compiler_spec) + def __init__(self, compiler_spec: CompilerSpec, arch: Optional[ArchSpec] = None) -> None: + err_msg = f"No compilers with spec {compiler_spec} found" if arch: - err_msg += " for operating system {0} and target {1}.".format(arch.os, arch.target) + err_msg += f" for operating system {arch.os} and target {arch.target}." super().__init__( err_msg, diff --git a/lib/spack/spack/environment/environment.py b/lib/spack/spack/environment/environment.py index b61332a0ab..7cf1057fa5 100644 --- a/lib/spack/spack/environment/environment.py +++ b/lib/spack/spack/environment/environment.py @@ -14,7 +14,7 @@ import stat import urllib.parse import urllib.request import warnings -from typing import Any, Dict, Iterable, List, Optional, Tuple, Union +from typing import Any, Dict, Iterable, List, Optional, Sequence, Tuple, Union import llnl.util.filesystem as fs import llnl.util.tty as tty @@ -55,7 +55,7 @@ from spack.spec import Spec from spack.spec_list import SpecList from spack.util.path import substitute_path_variables -SpecPair = Tuple[spack.spec.Spec, spack.spec.Spec] +SpecPair = spack.concretize.SpecPair #: environment variable used to indicate the active environment spack_env_var = "SPACK_ENV" @@ -1533,9 +1533,7 @@ class Environment: ] return new_user_specs, kept_user_specs, specs_to_concretize - def _concretize_together_where_possible( - self, tests: bool = False - ) -> List[Tuple[spack.spec.Spec, spack.spec.Spec]]: + def _concretize_together_where_possible(self, tests: bool = False) -> Sequence[SpecPair]: # Avoid cyclic dependency import spack.solver.asp @@ -1550,7 +1548,7 @@ class Environment: ret = [] result = spack.concretize.concretize_together_when_possible( - *specs_to_concretize, tests=tests + specs_to_concretize, tests=tests ) for abstract, concrete in result: # Only add to the environment if it's from this environment (not included in) @@ -1563,7 +1561,7 @@ class Environment: return ret - def _concretize_together(self, tests: bool = False) -> List[SpecPair]: + def _concretize_together(self, tests: bool = False) -> Sequence[SpecPair]: """Concretization strategy that concretizes all the specs in the same DAG. """ @@ -1577,8 +1575,8 @@ class Environment: self.specs_by_hash = {} try: - concretized_specs: List[SpecPair] = spack.concretize.concretize_together( - *specs_to_concretize, tests=tests + concretized_specs = spack.concretize.concretize_together( + specs_to_concretize, tests=tests ) except spack.error.UnsatisfiableSpecError as e: # "Enhance" the error message for multiple root specs, suggest a less strict @@ -1627,7 +1625,7 @@ class Environment: to_concretize = [ (root, None) for root in self.user_specs if root not in old_concretized_user_specs ] - concretized_specs = spack.concretize.concretize_separately(*to_concretize, tests=tests) + concretized_specs = spack.concretize.concretize_separately(to_concretize, tests=tests) by_hash = {} for abstract, concrete in concretized_specs: diff --git a/lib/spack/spack/test/concretize.py b/lib/spack/spack/test/concretize.py index 4d9940ea9b..bf96311d44 100644 --- a/lib/spack/spack/test/concretize.py +++ b/lib/spack/spack/test/concretize.py @@ -33,7 +33,6 @@ import spack.spec import spack.store import spack.util.file_cache import spack.variant as vt -from spack.concretize import find_spec from spack.installer import PackageInstaller from spack.spec import CompilerSpec, Spec from spack.version import Version, VersionList, ver @@ -674,39 +673,6 @@ class TestConcretize: assert spec["externaltool"].compiler.satisfies("gcc") assert spec["stuff"].compiler.satisfies("gcc") - def test_find_spec_parents(self): - """Tests the spec finding logic used by concretization.""" - s = Spec.from_literal({"a +foo": {"b +foo": {"c": None, "d+foo": None}, "e +foo": None}}) - - assert "a" == find_spec(s["b"], lambda s: "+foo" in s).name - - def test_find_spec_children(self): - s = Spec.from_literal({"a": {"b +foo": {"c": None, "d+foo": None}, "e +foo": None}}) - - assert "d" == find_spec(s["b"], lambda s: "+foo" in s).name - - s = Spec.from_literal({"a": {"b +foo": {"c+foo": None, "d": None}, "e +foo": None}}) - - assert "c" == find_spec(s["b"], lambda s: "+foo" in s).name - - def test_find_spec_sibling(self): - s = Spec.from_literal({"a": {"b +foo": {"c": None, "d": None}, "e +foo": None}}) - - assert "e" == find_spec(s["b"], lambda s: "+foo" in s).name - assert "b" == find_spec(s["e"], lambda s: "+foo" in s).name - - s = Spec.from_literal({"a": {"b +foo": {"c": None, "d": None}, "e": {"f +foo": None}}}) - - assert "f" == find_spec(s["b"], lambda s: "+foo" in s).name - - def test_find_spec_self(self): - s = Spec.from_literal({"a": {"b +foo": {"c": None, "d": None}, "e": None}}) - assert "b" == find_spec(s["b"], lambda s: "+foo" in s).name - - def test_find_spec_none(self): - s = Spec.from_literal({"a": {"b": {"c": None, "d": None}, "e": None}}) - assert find_spec(s["b"], lambda s: "+foo" in s) is None - def test_compiler_child(self): s = Spec("mpileaks%clang target=x86_64 ^dyninst%gcc") s.concretize() @@ -815,7 +781,7 @@ class TestConcretize: ) def test_simultaneous_concretization_of_specs(self, abstract_specs): abstract_specs = [Spec(x) for x in abstract_specs] - concrete_specs = spack.concretize.concretize_specs_together(*abstract_specs) + concrete_specs = spack.concretize.concretize_specs_together(abstract_specs) # Check there's only one configuration of each package in the DAG names = set(dep.name for spec in concrete_specs for dep in spec.traverse()) @@ -2137,7 +2103,7 @@ class TestConcretize: spack.config.set("packages", external_conf) abstract_specs = [Spec(s) for s in ["py-extension1", "python"]] - specs = spack.concretize.concretize_specs_together(*abstract_specs) + specs = spack.concretize.concretize_specs_together(abstract_specs) assert specs[0]["python"] == specs[1]["python"] @pytest.mark.regression("36190") |