diff --git a/src/cargo/util/semver_eval_ext.rs b/src/cargo/util/semver_eval_ext.rs index 41d1f10007a4..29448955d2bf 100644 --- a/src/cargo/util/semver_eval_ext.rs +++ b/src/cargo/util/semver_eval_ext.rs @@ -8,8 +8,16 @@ use semver::{Comparator, Op, Prerelease, Version, VersionReq}; pub(crate) fn matches_prerelease(req: &VersionReq, ver: &Version) -> bool { + // Whether there are pre release version can be as lower bound + let lower_bound_prerelease = &req.comparators.iter().any(|cmp| { + if matches!(cmp.op, Op::Greater | Op::GreaterEq) && !cmp.pre.is_empty() { + true + } else { + false + } + }); for cmp in &req.comparators { - if !matches_prerelease_impl(cmp, ver) { + if !matches_prerelease_impl(cmp, ver, lower_bound_prerelease) { return false; } } @@ -17,7 +25,7 @@ pub(crate) fn matches_prerelease(req: &VersionReq, ver: &Version) -> bool { true } -fn matches_prerelease_impl(cmp: &Comparator, ver: &Version) -> bool { +fn matches_prerelease_impl(cmp: &Comparator, ver: &Version, lower_bound_prerelease: &bool) -> bool { match cmp.op { Op::Exact | Op::Wildcard => matches_exact_prerelease(cmp, ver), Op::Greater => matches_greater(cmp, ver), @@ -27,7 +35,13 @@ fn matches_prerelease_impl(cmp: &Comparator, ver: &Version) -> bool { } matches_greater(cmp, ver) } - Op::Less => matches_less(&fill_partial_req(cmp), ver), + Op::Less => { + if *lower_bound_prerelease { + matches_less(&fill_partial_req(cmp), ver) + } else { + matches_less(&fill_partial_req_include_pre(cmp), ver) + } + } Op::LessEq => { if matches_exact_prerelease(cmp, ver) { return true; @@ -123,6 +137,21 @@ fn fill_partial_req(cmp: &Comparator) -> Comparator { cmp } +fn fill_partial_req_include_pre(cmp: &Comparator) -> Comparator { + let mut cmp = cmp.clone(); + if cmp.minor.is_none() { + cmp.minor = Some(0); + cmp.patch = Some(0); + cmp.pre = Prerelease::new("0").unwrap(); + } else if cmp.patch.is_none() { + cmp.patch = Some(0); + } + if cmp.pre.is_empty() { + cmp.pre = Prerelease::new("0").unwrap(); + } + cmp +} + fn matches_exact_prerelease(cmp: &Comparator, ver: &Version) -> bool { if matches_exact(cmp, ver) { return true; @@ -332,29 +361,66 @@ mod matches_prerelease_semantic { #[test] fn test_less() { - // 1.2.3, <2"), + req(">1.2.3, <2.0"), + req(">1.2.3, <2.0.0"), + req(">=1.2.3, <2.0.0"), + req(">1.2.3, <2.0.0-0"), + ] { + assert_match_all(r, &["1.2.4", "1.9.9"]); + assert_match_none(r, &["2.0.0-0", "2.0.0", "2.1.2"]); + } + + // Lower bound has prerelase tag, so upper bound doesn't change. + for r in &[ + req(">1.2.3-0, <2"), + req(">1.2.3-0, <2.0"), + req(">1.2.3-0, <2.0.0"), + req(">=1.2.3-0, <2.0.0"), + ] { + assert_match_all(r, &["1.2.4", "1.9.9", "2.0.0-0"]); + assert_match_none(r, &["2.0.0", "2.1.2"]); + } + + for r in &[ + req(">=2.0.0-0, <2"), + req(">=2.0.0-0, <2.0"), + req(">=2.0.0-0, <2.0.0"), + ] { + assert_match_all(r, &["2.0.0-0", "2.0.0-11"]); + assert_match_none(r, &["0.0.9", "2.0.0"]); + } + + // There is no intersection between lower bound and upper bound, in this case nothing matches + let ref r = req(">5.0.0, <2.0.0"); + assert_match_none(r, &["1.2.3", "3.0.0", "6.0.0"]); + let ref r = req(">5.0.0-0, <2.0.0"); + assert_match_none(r, &["1.2.3", "3.0.0", "6.0.0"]); + } + #[test] fn test_caret() { // ^I.J.K.0 (for I>0) — equivalent to >=I.J.K-0, <(I+1).0.0-0 diff --git a/src/cargo/util/semver_ext.rs b/src/cargo/util/semver_ext.rs index 462c749ef66d..d5321abde514 100644 --- a/src/cargo/util/semver_ext.rs +++ b/src/cargo/util/semver_ext.rs @@ -224,11 +224,11 @@ mod matches_prerelease { // (">1.2.3, <1.2.4", "1.2.3-0", false), (">1.2.3, <1.2.4", "1.2.3-1", false), - (">1.2.3, <1.2.4", "1.2.4-0", true), // upper bound semantic + (">1.2.3, <1.2.4", "1.2.4-0", false), // upper bound semantic // (">=1.2.3, <1.2.4", "1.2.3-0", false), (">=1.2.3, <1.2.4", "1.2.3-1", false), - (">=1.2.3, <1.2.4", "1.2.4-0", true), // upper bound semantic + (">=1.2.3, <1.2.4", "1.2.4-0", false), // upper bound semantic // (">1.2.3, <=1.2.4", "1.2.3-0", false), (">1.2.3, <=1.2.4", "1.2.3-1", false),