summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorMassimiliano Culpo <massimiliano.culpo@gmail.com>2023-09-21 17:02:34 +0200
committerGitHub <noreply@github.com>2023-09-21 17:02:34 +0200
commitabad16c1983685772f5c4ac36f1a9fecfe276793 (patch)
tree1662e39a4ac2b53fb2b9bac3f9962840a71ab389
parent4c0bc390544470b5308e70fd417e97a04e4fda69 (diff)
downloadspack-abad16c1983685772f5c4ac36f1a9fecfe276793.tar.gz
spack-abad16c1983685772f5c4ac36f1a9fecfe276793.tar.bz2
spack-abad16c1983685772f5c4ac36f1a9fecfe276793.tar.xz
spack-abad16c1983685772f5c4ac36f1a9fecfe276793.zip
Restore virtuals normalization on edge construction (#40130)
Put back normalization of the "virtuals" input as a sorted tuple. Without this we might get edges that differ just for the order of virtuals, or that have lists, which are not hashable. Add unit-tests to prevent regressions.
-rw-r--r--lib/spack/spack/spec.py2
-rw-r--r--lib/spack/spack/test/spec_semantics.py13
-rw-r--r--lib/spack/spack/test/spec_yaml.py5
3 files changed, 19 insertions, 1 deletions
diff --git a/lib/spack/spack/spec.py b/lib/spack/spack/spec.py
index c88998d54e..13bf66f506 100644
--- a/lib/spack/spack/spec.py
+++ b/lib/spack/spack/spec.py
@@ -740,7 +740,7 @@ class DependencySpec:
self.parent = parent
self.spec = spec
self.depflag = depflag
- self.virtuals = virtuals
+ self.virtuals = tuple(sorted(set(virtuals)))
def update_deptypes(self, depflag: dt.DepFlag) -> bool:
"""Update the current dependency types"""
diff --git a/lib/spack/spack/test/spec_semantics.py b/lib/spack/spack/test/spec_semantics.py
index 4f8ae8054a..662ea5ef0e 100644
--- a/lib/spack/spack/test/spec_semantics.py
+++ b/lib/spack/spack/test/spec_semantics.py
@@ -11,6 +11,7 @@ from spack.error import SpecError, UnsatisfiableSpecError
from spack.spec import (
ArchSpec,
CompilerSpec,
+ DependencySpec,
Spec,
SpecFormatSigilError,
SpecFormatStringError,
@@ -1326,3 +1327,15 @@ def test_abstract_hash_intersects_and_satisfies(default_mock_concretization):
# disjoint concretization space
assert_disjoint(abstract_none, concrete)
assert_disjoint(abstract_none, abstract_5)
+
+
+def test_edge_equality_does_not_depend_on_virtual_order():
+ """Tests that two edges that are constructed with just a different order of the virtuals in
+ the input parameters are equal to each other.
+ """
+ parent, child = Spec("parent"), Spec("child")
+ edge1 = DependencySpec(parent, child, depflag=0, virtuals=("mpi", "lapack"))
+ edge2 = DependencySpec(parent, child, depflag=0, virtuals=("lapack", "mpi"))
+ assert edge1 == edge2
+ assert tuple(sorted(edge1.virtuals)) == edge1.virtuals
+ assert tuple(sorted(edge2.virtuals)) == edge1.virtuals
diff --git a/lib/spack/spack/test/spec_yaml.py b/lib/spack/spack/test/spec_yaml.py
index 3599606f1c..b19bb4308f 100644
--- a/lib/spack/spack/test/spec_yaml.py
+++ b/lib/spack/spack/test/spec_yaml.py
@@ -500,3 +500,8 @@ def test_load_json_specfiles(specfile, expected_hash, reader_cls):
openmpi_edges = s2.edges_to_dependencies(name="openmpi")
assert len(openmpi_edges) == 1
+
+ # The virtuals attribute must be a tuple, when read from a
+ # JSON or YAML file, not a list
+ for edge in s2.traverse_edges():
+ assert isinstance(edge.virtuals, tuple), edge