Source code for spack.hash_lookup

# Copyright Spack Project Developers. See COPYRIGHT file for details.
#
# SPDX-License-Identifier: (Apache-2.0 OR MIT)
"""Resolve abstract hash references in Specs to concrete specs.

This module is free of spack.solver imports. It locates matching specs by
searching the active environment, the installed store, the binary cache, and
configured externals (via spack.externals_config).
"""

from typing import List

import spack.binary_distribution
import spack.config
import spack.environment
import spack.error
import spack.externals_config
import spack.spec
import spack.store
from spack.enums import InstallRecordStatus


def _matching_external_specs(spec: "spack.spec.Spec") -> List["spack.spec.Spec"]:
    """Return configured externals from packages.yaml that match spec by abstract hash."""
    config = spack.config.CONFIG
    try:
        packages_with_externals = spack.externals_config.external_config_with_implicit_externals(
            config
        )
        completion_mode = config.get("concretizer:externals:completion")
        parser = spack.externals_config.create_external_parser(
            packages_with_externals, completion_mode
        )
    except spack.error.SpackError:
        return []
    return parser.query(spec)


def _lookup_one(spec: "spack.spec.Spec") -> "spack.spec.Spec":
    """Return the single concrete spec matching an abstract-hash spec.

    Searches in order: active environment, configured externals, installed store, binary cache.
    Raises InvalidHashError if nothing matches, AmbiguousHashError if more than one matches.
    """
    active_env = spack.environment.active_environment()

    matches = (
        (active_env.all_matching_specs(spec) if active_env else [])
        or _matching_external_specs(spec)
        or spack.store.STORE.db.query(spec, installed=InstallRecordStatus.ANY)
        or spack.binary_distribution.BinaryCacheQuery(True)(spec)
    )

    if not matches:
        raise spack.spec.InvalidHashError(spec, spec.abstract_hash)

    if len(matches) != 1:
        raise spack.spec.AmbiguousHashError(
            f"Multiple packages specify hash beginning '{spec.abstract_hash}'.", *matches
        )

    return matches[0]


[docs] def lookup_hash(spec: "spack.spec.Spec") -> "spack.spec.Spec": """Return a copy of spec with all abstract-hash nodes replaced by their concrete counterparts. Non-destructive: always returns a new Spec object. If spec is already concrete or has no abstract-hash nodes, returns spec unchanged. """ if spec.concrete or not any(node.abstract_hash for node in spec.traverse()): return spec result = spec.copy(deps=False) if result.abstract_hash: result._dup(_lookup_one(spec)) return result node_lookup = { id(node): _lookup_one(node) for node in spec.traverse(root=False) if node.abstract_hash } for edge in spec.traverse_edges(root=False): key = edge.parent.name current_node = result if key == result.name else result[key] child_node = node_lookup.get(id(edge.spec), edge.spec.copy()) current_node._add_dependency( child_node, depflag=edge.depflag, virtuals=edge.virtuals, direct=edge.direct ) return result
[docs] def replace_hash(spec: "spack.spec.Spec") -> None: """Populate spec in-place by resolving all abstract-hash nodes. Destructive counterpart to lookup_hash. No-op if spec has no abstract-hash nodes. """ if not any(node for node in spec.traverse(order="post") if node.abstract_hash): return spec._dup(lookup_hash(spec))