From 7da06153da2528fd7f708e7e67175c8b240f4b77 Mon Sep 17 00:00:00 2001 From: Zanie Blue Date: Mon, 5 Feb 2024 08:36:10 -0600 Subject: [PATCH] Add a `reason` to the `NoVersions` incompatibility (#22) * Add a `reason` to the `NoVersions` incompatibility * Make optional --- examples/unsat_root_message_no_version.rs | 2 +- src/internal/incompatibility.rs | 29 +++++++++++++---------- src/report.rs | 18 +++++++------- src/solver.rs | 3 ++- 4 files changed, 29 insertions(+), 23 deletions(-) diff --git a/examples/unsat_root_message_no_version.rs b/examples/unsat_root_message_no_version.rs index 6e133129..80acce7c 100644 --- a/examples/unsat_root_message_no_version.rs +++ b/examples/unsat_root_message_no_version.rs @@ -66,7 +66,7 @@ impl ReportFormatter> for CustomReportFormatter External::NotRoot(package, version) => { format!("we are solving dependencies of {package} {version}") } - External::NoVersions(package, set) => { + External::NoVersions(package, set, _) => { if set == &Range::full() { format!("there is no available version for {package}") } else { diff --git a/src/internal/incompatibility.rs b/src/internal/incompatibility.rs index dfc8038c..629582d9 100644 --- a/src/internal/incompatibility.rs +++ b/src/internal/incompatibility.rs @@ -46,8 +46,8 @@ pub type IncompId = Id>; pub enum Kind { /// Initial incompatibility aiming at picking the root package for the first decision. NotRoot(P, VS::V), - /// There are no versions in the given range for this package. - NoVersions(P, VS), + /// There are no versions in the given range for this package. A string reason may be included. + NoVersions(P, VS, Option), /// The package is unavailable for versions in the range. A string reason is included. Unavailable(P, VS, String), /// Incompatibility coming from the dependencies of a given package. @@ -87,14 +87,14 @@ impl Incompatibility { /// Create an incompatibility to remember /// that a given set does not contain any version. - pub fn no_versions(package: P, term: Term) -> Self { + pub fn no_versions(package: P, term: Term, reason: Option) -> Self { let set = match &term { Term::Positive(r) => r.clone(), Term::Negative(_) => panic!("No version should have a positive term"), }; Self { package_terms: SmallMap::One([(package.clone(), term)]), - kind: Kind::NoVersions(package, set), + kind: Kind::NoVersions(package, set, reason), } } @@ -254,15 +254,20 @@ impl Incompatibility { Kind::NotRoot(package, version) => { DerivationTree::External(External::NotRoot(package, version)) } - Kind::NoVersions(package, set) => { - DerivationTree::External(External::NoVersions(package, set)) - } - Kind::Unavailable(package, set, reason) => { - DerivationTree::External(External::Unavailable(package, set, reason)) - } - Kind::FromDependencyOf(package, set, dep_package, dep_set) => DerivationTree::External( - External::FromDependencyOf(package, set, dep_package, dep_set), + Kind::NoVersions(package, set, reason) => DerivationTree::External( + External::NoVersions(package.clone(), set.clone(), reason.clone()), ), + Kind::Unavailable(package, set, reason) => DerivationTree::External( + External::Unavailable(package.clone(), set.clone(), reason.clone()), + ), + Kind::FromDependencyOf(package, set, dep_package, dep_set) => { + DerivationTree::External(External::FromDependencyOf( + package.clone(), + set.clone(), + dep_package.clone(), + dep_set.clone(), + )) + } } } } diff --git a/src/report.rs b/src/report.rs index e7a01981..5f5f6eeb 100644 --- a/src/report.rs +++ b/src/report.rs @@ -47,8 +47,8 @@ pub enum DerivationTree { pub enum External { /// Initial incompatibility aiming at picking the root package for the first decision. NotRoot(P, VS::V), - /// There are no versions in the given set for this package. - NoVersions(P, VS), + /// There are no versions in the given set for this package. A string reason is included. + NoVersions(P, VS, Option), /// The package is unusable in the given set. A string reason is included. Unavailable(P, VS, String), /// Incompatibility coming from the dependencies of a given package. @@ -82,7 +82,7 @@ impl DerivationTree { packages.insert(p); packages.insert(p2); } - External::NoVersions(p, _) + External::NoVersions(p, _, _) | External::NotRoot(p, _) | External::Unavailable(p, ..) => { packages.insert(p); @@ -113,14 +113,14 @@ impl DerivationTree { Arc::make_mut(&mut derived.cause1), Arc::make_mut(&mut derived.cause2), ) { - (DerivationTree::External(External::NoVersions(p, r)), ref mut cause2) => { + (DerivationTree::External(External::NoVersions(p, r, _)), ref mut cause2) => { cause2.collapse_no_versions(); *self = cause2 .clone() .merge_no_versions(p.to_owned(), r.to_owned()) .unwrap_or_else(|| self.to_owned()); } - (ref mut cause1, DerivationTree::External(External::NoVersions(p, r))) => { + (ref mut cause1, DerivationTree::External(External::NoVersions(p, r, _))) => { cause1.collapse_no_versions(); *self = cause1 .clone() @@ -144,9 +144,9 @@ impl DerivationTree { DerivationTree::External(External::NotRoot(_, _)) => { panic!("How did we end up with a NoVersions merged with a NotRoot?") } - DerivationTree::External(External::NoVersions(_, r)) => Some(DerivationTree::External( - External::NoVersions(package, set.union(&r)), - )), + // + // Cannot be merged because the reason may not match + DerivationTree::External(External::NoVersions(_, _, _)) => None, // Cannot be merged because the reason may not match DerivationTree::External(External::Unavailable(_, _, _)) => None, DerivationTree::External(External::FromDependencyOf(p1, r1, p2, r2)) => { @@ -176,7 +176,7 @@ impl fmt::Display for External { Self::NotRoot(package, version) => { write!(f, "we are solving dependencies of {} {}", package, version) } - Self::NoVersions(package, set) => { + Self::NoVersions(package, set, _) => { if set == &VS::full() { write!(f, "there is no available version for {}", package) } else { diff --git a/src/solver.rs b/src/solver.rs index f17ad8e0..885812a5 100644 --- a/src/solver.rs +++ b/src/solver.rs @@ -120,7 +120,8 @@ pub fn resolve( // Pick the next compatible version. let v = match decision { None => { - let inc = Incompatibility::no_versions(next.clone(), term_intersection.clone()); + let inc = + Incompatibility::no_versions(next.clone(), term_intersection.clone(), None); state.add_incompatibility(inc); continue; }