Source code for spack.concretize

# Copyright Spack Project Developers. See COPYRIGHT file for details.
#
# SPDX-License-Identifier: (Apache-2.0 OR MIT)
"""High-level functions to concretize list of specs"""

import importlib
import sys
import time
from typing import TYPE_CHECKING, Dict, Iterable, List, Optional, Sequence, Tuple, Union

import spack.compilers
import spack.compilers.config
import spack.config
import spack.error
import spack.llnl.util.tty as tty
import spack.repo
import spack.util.parallel
from spack.spec import ArchSpec, CompilerSpec, Spec

SpecPairInput = Tuple[Spec, Optional[Spec]]
SpecPair = Tuple[Spec, Spec]
TestsType = Union[bool, Iterable[str]]

if TYPE_CHECKING:
    from spack.solver.reuse import SpecFiltersFactory


def _concretize_specs_together(
    abstract_specs: Sequence[Spec],
    *,
    tests: TestsType = False,
    factory: Optional["SpecFiltersFactory"] = None,
) -> List[Spec]:
    """Given a number of specs as input, tries to concretize them together.

    Args:
        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.
        factory: optional factory to produce a list of specs to be reused
    """
    from spack.solver.asp import Solver

    allow_deprecated = spack.config.get("config:deprecated", False)
    result = Solver(specs_factory=factory).solve(
        abstract_specs, tests=tests, allow_deprecated=allow_deprecated
    )
    return [s.copy() for s in result.specs]


[docs] def concretize_together( spec_list: Sequence[SpecPairInput], *, tests: TestsType = False, factory: Optional["SpecFiltersFactory"] = None, ) -> List[SpecPair]: """Given a number of specs as input, tries to concretize them together. Args: spec_list: list of tuples to concretize. First entry is abstract spec, second entry is already concrete spec or None if not yet 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. factory: optional factory to produce a list of specs to be reused """ 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, tests=tests, factory=factory) return list(zip(abstract_specs, concrete_specs))
[docs] def concretize_together_when_possible( spec_list: Sequence[SpecPairInput], *, tests: TestsType = False, factory: Optional["SpecFiltersFactory"] = None, ) -> List[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: spec_list: list of tuples to concretize. First entry is abstract spec, second entry is already concrete spec or None if not yet 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. factory: optional factory to produce a list of specs to be reused """ from spack.solver.asp import Solver to_concretize = [concrete if concrete else abstract for abstract, concrete in spec_list] old_concrete_to_abstract = { concrete: abstract for (abstract, concrete) in spec_list if concrete } result_by_user_spec: Dict[Spec, Spec] = {} allow_deprecated = spack.config.get("config:deprecated", False) j = 0 start = time.monotonic() for result in Solver(specs_factory=factory).solve_in_rounds( to_concretize, tests=tests, allow_deprecated=allow_deprecated ): now = time.monotonic() duration = now - start percentage = int((j + 1) / len(to_concretize) * 100) for abstract, concrete in result.specs_by_input.items(): tty.verbose( f"{duration:6.1f}s [{percentage:3d}%] {concrete.cformat('{hash:7}')} " f"{abstract.colored_str}" ) j += 1 sys.stdout.flush() result_by_user_spec.update(result.specs_by_input) start = now # If the "abstract" spec is a concrete spec from the previous concretization # translate it back to an abstract spec. Otherwise, keep the abstract spec return [ (old_concrete_to_abstract.get(abstract, abstract), concrete) for abstract, concrete in sorted(result_by_user_spec.items()) ]
[docs] def concretize_separately( spec_list: Sequence[SpecPairInput], *, tests: TestsType = False, factory: Optional["SpecFiltersFactory"] = None, ) -> List[SpecPair]: """Concretizes the input specs separately from each other. Args: spec_list: list of tuples to concretize. First entry is abstract spec, second entry is already concrete spec or None if not yet 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. factory: optional factory to produce a list of specs to be reused """ from spack.bootstrap import ( ensure_bootstrap_configuration, ensure_clingo_importable_or_raise, ensure_winsdk_external_or_raise, ) to_concretize = [abstract for abstract, concrete in spec_list if not concrete] args = [ (i, str(abstract), tests, factory) for i, abstract in enumerate(to_concretize) if not abstract.concrete ] ret = [(i, abstract) for i, abstract in enumerate(to_concretize) if abstract.concrete] try: # Ensure we don't try to bootstrap clingo in parallel importlib.import_module("clingo") except ImportError: with ensure_bootstrap_configuration(): ensure_clingo_importable_or_raise() # ensure we don't try to detect winsdk in parallel if sys.platform == "win32": ensure_winsdk_external_or_raise() # Ensure all the indexes have been built or updated, since # otherwise the processes in the pool may timeout on waiting # for a write lock. We do this indirectly by retrieving the # provider index, which should in turn trigger the update of # all the indexes if there's any need for that. _ = spack.repo.PATH.provider_index # Ensure we have compilers in packages.yaml to avoid that # processes try to write the config file in parallel _ = spack.compilers.config.all_compilers() # Early return if there is nothing to do if len(args) == 0: # Still have to combine the things that were passed in as abstract with the things # that were passed in as pairs return [(abstract, concrete) for abstract, (_, concrete) in zip(to_concretize, ret)] + [ (abstract, concrete) for abstract, concrete in spec_list if concrete ] # Solve the environment in parallel on Linux num_procs = min(len(args), spack.config.determine_number_of_jobs(parallel=True)) msg = "Starting concretization" # no parallel conc on Windows if not sys.platform == "win32" and num_procs > 1: msg += f" pool with {num_procs} processes" tty.msg(msg) for j, (i, concrete, duration) in enumerate( spack.util.parallel.imap_unordered( _concretize_task, args, processes=num_procs, debug=tty.is_debug(), maxtaskperchild=1 ) ): ret.append((i, concrete)) percentage = int((j + 1) / len(args) * 100) tty.verbose( f"{duration:6.1f}s [{percentage:3d}%] {concrete.cformat('{hash:7}')} " f"{to_concretize[i].colored_str}" ) sys.stdout.flush() # Add specs in original order ret.sort(key=lambda x: x[0]) return [(abstract, concrete) for abstract, (_, concrete) in zip(to_concretize, ret)] + [ (abstract, concrete) for abstract, concrete in spec_list if concrete ]
def _concretize_task( packed_arguments: Tuple[int, str, TestsType, Optional["SpecFiltersFactory"]], ) -> Tuple[int, Spec, float]: index, spec_str, tests, factory = packed_arguments with tty.SuppressOutput(msg_enabled=False): start = time.time() spec = concretize_one(Spec(spec_str), tests=tests, factory=factory) return index, spec, time.time() - start
[docs] def concretize_one( spec: Union[str, Spec], *, tests: TestsType = False, factory: Optional["SpecFiltersFactory"] = None, ) -> Spec: """Return a concretized copy of the given spec. Args: tests: if False disregard test dependencies, if a list of names activate them for the packages in the list, if True activate test dependencies for all packages. """ from spack.solver.asp import Solver, SpecBuilder if isinstance(spec, str): spec = Spec(spec) spec = spec.lookup_hash() if spec.concrete: return spec.copy() for node in spec.traverse(): if not node.name: raise spack.error.SpecError( f"Spec {node} has no name; cannot concretize an anonymous spec" ) allow_deprecated = spack.config.get("config:deprecated", False) result = Solver(specs_factory=factory).solve( [spec], tests=tests, allow_deprecated=allow_deprecated ) # take the best answer opt, i, answer = min(result.answers) name = spec.name # TODO: Consolidate this code with similar code in solve.py if spack.repo.PATH.is_virtual(spec.name): providers = [s.name for s in answer.values() if s.package.provides(name)] name = providers[0] node = SpecBuilder.make_node(pkg=name) assert node in answer, ( f"cannot find {name} in the list of specs {','.join([n.pkg for n in answer.keys()])}" ) concretized = answer[node] return concretized
[docs] class UnavailableCompilerVersionError(spack.error.SpackError): """Raised when there is no available compiler that satisfies a 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 += f" for operating system {arch.os} and target {arch.target}." super().__init__( err_msg, "Run 'spack compiler find' to add compilers or " "'spack compilers' to see which compilers are already recognized" " by spack.", )