use std::error::Error;
use std::fmt;
use std::str;
use semver_parser;
use semver_parser::{Compat, RangeSet};
use version::Identifier;
use Version;
#[cfg(feature = "serde")]
use serde::de::{self, Deserialize, Deserializer, Visitor};
#[cfg(feature = "serde")]
use serde::ser::{Serialize, Serializer};
use self::Op::{Ex, Gt, GtEq, Lt, LtEq};
use self::ReqParseError::*;
#[derive(Clone, PartialEq, Eq, PartialOrd, Ord, Hash, Debug)]
#[cfg_attr(feature = "diesel", derive(AsExpression, FromSqlRow))]
#[cfg_attr(feature = "diesel", sql_type = "diesel::sql_types::Text")]
pub struct VersionReq {
ranges: Vec<Range>,
compat: Compat,
}
impl From<semver_parser::RangeSet> for VersionReq {
fn from(range_set: semver_parser::RangeSet) -> VersionReq {
VersionReq {
ranges: range_set.ranges.into_iter().map(From::from).collect(),
compat: range_set.compat,
}
}
}
#[cfg(feature = "serde")]
impl Serialize for VersionReq {
fn serialize<S>(&self, serializer: S) -> ::std::result::Result<S::Ok, S::Error>
where
S: Serializer,
{
serializer.collect_str(self)
}
}
#[cfg(feature = "serde")]
impl<'de> Deserialize<'de> for VersionReq {
fn deserialize<D>(deserializer: D) -> ::std::result::Result<Self, D::Error>
where
D: Deserializer<'de>,
{
struct VersionReqVisitor;
impl<'de> Visitor<'de> for VersionReqVisitor {
type Value = VersionReq;
fn expecting(&self, formatter: &mut fmt::Formatter) -> fmt::Result {
formatter.write_str("a SemVer version requirement as a string")
}
fn visit_str<E>(self, v: &str) -> ::std::result::Result<Self::Value, E>
where
E: de::Error,
{
VersionReq::parse(v).map_err(de::Error::custom)
}
}
deserializer.deserialize_str(VersionReqVisitor)
}
}
#[derive(Clone, PartialEq, Eq, PartialOrd, Ord, Hash, Debug)]
enum Op {
Ex,
Gt,
GtEq,
Lt,
LtEq,
}
impl From<semver_parser::Op> for Op {
fn from(op: semver_parser::Op) -> Op {
match op {
semver_parser::Op::Eq => Op::Ex,
semver_parser::Op::Gt => Op::Gt,
semver_parser::Op::Gte => Op::GtEq,
semver_parser::Op::Lt => Op::Lt,
semver_parser::Op::Lte => Op::LtEq,
}
}
}
#[derive(Clone, PartialEq, Eq, PartialOrd, Ord, Hash, Debug)]
struct Range {
predicates: Vec<Predicate>,
compat: Compat,
}
impl From<semver_parser::Range> for Range {
fn from(range: semver_parser::Range) -> Range {
Range {
predicates: range.comparator_set.into_iter().map(From::from).collect(),
compat: range.compat,
}
}
}
#[derive(Clone, PartialEq, Eq, PartialOrd, Ord, Hash, Debug)]
struct Predicate {
op: Op,
major: u64,
minor: u64,
patch: u64,
pre: Vec<Identifier>,
}
impl From<semver_parser::Comparator> for Predicate {
fn from(comparator: semver_parser::Comparator) -> Predicate {
Predicate {
op: From::from(comparator.op),
major: comparator.major,
minor: comparator.minor,
patch: comparator.patch,
pre: comparator.pre.into_iter().map(From::from).collect(),
}
}
}
impl From<semver_parser::Identifier> for Identifier {
fn from(identifier: semver_parser::Identifier) -> Identifier {
match identifier {
semver_parser::Identifier::Numeric(n) => Identifier::Numeric(n),
semver_parser::Identifier::AlphaNumeric(s) => Identifier::AlphaNumeric(s),
}
}
}
#[derive(Clone, Debug, PartialEq)]
pub enum ReqParseError {
InvalidVersionRequirement,
OpAlreadySet,
InvalidSigil,
VersionComponentsMustBeNumeric,
InvalidIdentifier,
MajorVersionRequired,
UnimplementedVersionRequirement,
DeprecatedVersionRequirement(VersionReq),
}
impl fmt::Display for ReqParseError {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
let msg = match self {
InvalidVersionRequirement => "the given version requirement is invalid",
OpAlreadySet => {
"you have already provided an operation, such as =, ~, or ^; only use one"
}
InvalidSigil => "the sigil you have written is not correct",
VersionComponentsMustBeNumeric => "version components must be numeric",
InvalidIdentifier => "invalid identifier",
MajorVersionRequired => "at least a major version number is required",
UnimplementedVersionRequirement => {
"the given version requirement is not implemented, yet"
}
DeprecatedVersionRequirement(_) => "This requirement is deprecated",
};
msg.fmt(f)
}
}
impl Error for ReqParseError {}
impl From<String> for ReqParseError {
fn from(other: String) -> ReqParseError {
match &*other {
"Null is not a valid VersionReq" => ReqParseError::InvalidVersionRequirement,
"VersionReq did not parse properly." => ReqParseError::OpAlreadySet,
_ => ReqParseError::InvalidVersionRequirement,
}
}
}
impl VersionReq {
pub fn any() -> VersionReq {
VersionReq {
ranges: vec![],
compat: Compat::Cargo,
}
}
pub fn parse(input: &str) -> Result<VersionReq, ReqParseError> {
let range_set = input.parse::<RangeSet>();
if let Ok(v) = range_set {
return Ok(From::from(v));
}
match VersionReq::parse_deprecated(input) {
Some(v) => Err(ReqParseError::DeprecatedVersionRequirement(v)),
None => Err(From::from(range_set.err().unwrap())),
}
}
pub fn parse_compat(input: &str, compat: Compat) -> Result<VersionReq, ReqParseError> {
let range_set = RangeSet::parse(input, compat);
if let Ok(v) = range_set {
return Ok(From::from(v));
}
match VersionReq::parse_deprecated(input) {
Some(v) => Err(ReqParseError::DeprecatedVersionRequirement(v)),
None => Err(From::from(range_set.err().unwrap())),
}
}
fn parse_deprecated(version: &str) -> Option<VersionReq> {
match version {
".*" => Some(VersionReq::any()),
"0.1.0." => Some(VersionReq::parse("0.1.0").unwrap()),
"0.3.1.3" => Some(VersionReq::parse("0.3.13").unwrap()),
"0.2*" => Some(VersionReq::parse("0.2.*").unwrap()),
"*.0" => Some(VersionReq::any()),
_ => None,
}
}
pub fn exact(version: &Version) -> VersionReq {
VersionReq {
ranges: vec![Range {
predicates: vec![Predicate::exact(version)],
compat: Compat::Cargo,
}],
compat: Compat::Cargo,
}
}
pub fn matches(&self, version: &Version) -> bool {
if self.ranges.is_empty() {
return true;
}
self.ranges
.iter()
.any(|r| r.matches(version) && r.pre_tag_is_compatible(version))
}
pub fn is_exact(&self) -> bool {
if let [range] = self.ranges.as_slice() {
if let [predicate] = range.predicates.as_slice() {
return predicate.has_exactly_one_match();
}
}
false
}
}
impl str::FromStr for VersionReq {
type Err = ReqParseError;
fn from_str(s: &str) -> Result<VersionReq, ReqParseError> {
VersionReq::parse(s)
}
}
impl Range {
fn matches(&self, ver: &Version) -> bool {
self.predicates.iter().all(|p| p.matches(ver))
}
fn pre_tag_is_compatible(&self, ver: &Version) -> bool {
self.predicates.iter().any(|p| p.pre_tag_is_compatible(ver))
}
}
impl Predicate {
fn exact(version: &Version) -> Predicate {
Predicate {
op: Ex,
major: version.major,
minor: version.minor,
patch: version.patch,
pre: version.pre.clone(),
}
}
pub fn matches(&self, ver: &Version) -> bool {
match self.op {
Ex => self.matches_exact(ver),
Gt => self.matches_greater(ver),
GtEq => self.matches_exact(ver) || self.matches_greater(ver),
Lt => !self.matches_exact(ver) && !self.matches_greater(ver),
LtEq => !self.matches_greater(ver),
}
}
fn matches_exact(&self, ver: &Version) -> bool {
self.major == ver.major
&& self.minor == ver.minor
&& self.patch == ver.patch
&& self.pre == ver.pre
}
fn pre_tag_is_compatible(&self, ver: &Version) -> bool {
!ver.is_prerelease()
|| (self.major == ver.major
&& self.minor == ver.minor
&& self.patch == ver.patch
&& !self.pre.is_empty())
}
fn matches_greater(&self, ver: &Version) -> bool {
if self.major != ver.major {
return ver.major > self.major;
}
if self.minor != ver.minor {
return ver.minor > self.minor;
}
if self.patch != ver.patch {
return ver.patch > self.patch;
}
if !self.pre.is_empty() {
return ver.pre.is_empty() || ver.pre > self.pre;
}
false
}
fn has_exactly_one_match(&self) -> bool {
self.op == Ex
}
}
impl fmt::Display for VersionReq {
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
if self.ranges.is_empty() {
write!(fmt, "*")?;
} else {
for (i, ref pred) in self.ranges.iter().enumerate() {
if i == 0 {
write!(fmt, "{}", pred)?;
} else {
write!(fmt, " || {}", pred)?;
}
}
}
Ok(())
}
}
impl fmt::Display for Range {
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
for (i, ref pred) in self.predicates.iter().enumerate() {
if i == 0 {
write!(fmt, "{}", pred)?;
} else if self.compat == Compat::Npm {
write!(fmt, " {}", pred)?;
} else {
write!(fmt, ", {}", pred)?;
}
}
Ok(())
}
}
impl fmt::Display for Predicate {
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
write!(
fmt,
"{}{}.{}.{}",
self.op, self.major, self.minor, self.patch
)?;
if !self.pre.is_empty() {
write!(fmt, "-")?;
for (i, x) in self.pre.iter().enumerate() {
if i != 0 {
write!(fmt, ".")?
}
write!(fmt, "{}", x)?;
}
}
Ok(())
}
}
impl fmt::Display for Op {
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
match *self {
Ex => write!(fmt, "=")?,
Gt => write!(fmt, ">")?,
GtEq => write!(fmt, ">=")?,
Lt => write!(fmt, "<")?,
LtEq => write!(fmt, "<=")?,
}
Ok(())
}
}
#[cfg(test)]
mod test {
use super::super::version::Version;
use super::{Compat, Op, VersionReq};
use std::hash::{Hash, Hasher};
fn req(s: &str) -> VersionReq {
VersionReq::parse(s).unwrap()
}
fn req_npm(s: &str) -> VersionReq {
VersionReq::parse_compat(s, Compat::Npm).unwrap()
}
fn version(s: &str) -> Version {
match Version::parse(s) {
Ok(v) => v,
Err(e) => panic!("`{}` is not a valid version. Reason: {:?}", s, e),
}
}
fn assert_match(req: &VersionReq, vers: &[&str]) {
for ver in vers.iter() {
assert!(req.matches(&version(*ver)), "did not match {}", ver);
}
}
fn assert_not_match(req: &VersionReq, vers: &[&str]) {
for ver in vers.iter() {
assert!(!req.matches(&version(*ver)), "matched {}", ver);
}
}
fn calculate_hash<T: Hash>(t: T) -> u64 {
use std::collections::hash_map::DefaultHasher;
let mut s = DefaultHasher::new();
t.hash(&mut s);
s.finish()
}
#[test]
fn test_parsing_default() {
let r = req("1.0.0");
assert_eq!(r.to_string(), ">=1.0.0, <2.0.0".to_string());
assert_match(&r, &["1.0.0", "1.0.1"]);
assert_not_match(&r, &["0.9.9", "0.10.0", "0.1.0"]);
}
#[test]
fn test_parsing_default_npm() {
let r = req_npm("1.0.0");
assert_eq!(r.to_string(), "=1.0.0".to_string());
assert_match(&r, &["1.0.0"]);
assert_not_match(&r, &["0.9.9", "0.10.0", "0.1.0", "1.0.1"]);
}
#[test]
fn test_parsing_exact() {
let r = req("=1.0.0");
assert!(r.to_string() == "=1.0.0".to_string());
assert_eq!(r.to_string(), "=1.0.0".to_string());
assert_match(&r, &["1.0.0"]);
assert_not_match(&r, &["1.0.1", "0.9.9", "0.10.0", "0.1.0", "1.0.0-pre"]);
let r = req("=0.9.0");
assert_eq!(r.to_string(), "=0.9.0".to_string());
assert_match(&r, &["0.9.0"]);
assert_not_match(&r, &["0.9.1", "1.9.0", "0.0.9"]);
let r = req("=0.1.0-beta2.a");
assert_eq!(r.to_string(), "=0.1.0-beta2.a".to_string());
assert_match(&r, &["0.1.0-beta2.a"]);
assert_not_match(&r, &["0.9.1", "0.1.0", "0.1.1-beta2.a", "0.1.0-beta2"]);
}
#[test]
fn test_parse_metadata_see_issue_88_see_issue_88() {
for op in &[Op::Ex, Op::Gt, Op::GtEq, Op::Lt, Op::LtEq] {
println!("{} 1.2.3+meta", op);
req(&format!("{} 1.2.3+meta", op));
}
}
#[test]
pub fn test_parsing_greater_than() {
let r = req(">= 1.0.0");
assert_eq!(r.to_string(), ">=1.0.0".to_string());
assert_match(&r, &["1.0.0", "2.0.0"]);
assert_not_match(&r, &["0.1.0", "0.0.1", "1.0.0-pre", "2.0.0-pre"]);
let r = req(">= 2.1.0-alpha2");
assert_match(&r, &["2.1.0-alpha2", "2.1.0-alpha3", "2.1.0", "3.0.0"]);
assert_not_match(
&r,
&["2.0.0", "2.1.0-alpha1", "2.0.0-alpha2", "3.0.0-alpha2"],
);
}
#[test]
pub fn test_parsing_less_than() {
let r = req("< 1.0.0");
assert_eq!(r.to_string(), "<1.0.0".to_string());
assert_match(&r, &["0.1.0", "0.0.1"]);
assert_not_match(&r, &["1.0.0", "1.0.0-beta", "1.0.1", "0.9.9-alpha"]);
let r = req("<= 2.1.0-alpha2");
assert_match(&r, &["2.1.0-alpha2", "2.1.0-alpha1", "2.0.0", "1.0.0"]);
assert_not_match(
&r,
&["2.1.0", "2.2.0-alpha1", "2.0.0-alpha2", "1.0.0-alpha2"],
);
}
#[test]
pub fn test_multiple() {
let r = req("> 0.0.9, <= 2.5.3");
assert_eq!(r.to_string(), ">0.0.9, <=2.5.3".to_string());
assert_match(&r, &["0.0.10", "1.0.0", "2.5.3"]);
assert_not_match(&r, &["0.0.8", "2.5.4"]);
let r = req("0.3.0, 0.4.0");
assert_eq!(
r.to_string(),
">=0.3.0, <0.4.0, >=0.4.0, <0.5.0".to_string()
);
assert_not_match(&r, &["0.0.8", "0.3.0", "0.4.0"]);
let r = req("<= 0.2.0, >= 0.5.0");
assert_eq!(r.to_string(), "<=0.2.0, >=0.5.0".to_string());
assert_not_match(&r, &["0.0.8", "0.3.0", "0.5.1"]);
let r = req("0.1.0, 0.1.4, 0.1.6");
assert_eq!(
r.to_string(),
">=0.1.0, <0.2.0, >=0.1.4, <0.2.0, >=0.1.6, <0.2.0".to_string()
);
assert_match(&r, &["0.1.6", "0.1.9"]);
assert_not_match(&r, &["0.1.0", "0.1.4", "0.2.0"]);
assert!(VersionReq::parse("> 0.1.0,").is_err());
assert!(VersionReq::parse("> 0.3.0, ,").is_err());
let r = req(">=0.5.1-alpha3, <0.6");
assert_eq!(r.to_string(), ">=0.5.1-alpha3, <0.6.0".to_string());
assert_match(
&r,
&[
"0.5.1-alpha3",
"0.5.1-alpha4",
"0.5.1-beta",
"0.5.1",
"0.5.5",
],
);
assert_not_match(
&r,
&["0.5.1-alpha1", "0.5.2-alpha3", "0.5.5-pre", "0.5.0-pre"],
);
assert_not_match(&r, &["0.6.0", "0.6.0-pre"]);
let r = req("1.2.3 - 2.3.4");
assert_eq!(r.to_string(), ">=1.2.3, <=2.3.4");
assert_match(&r, &["1.2.3", "1.2.10", "2.0.0", "2.3.4"]);
assert_not_match(&r, &["1.0.0", "1.2.2", "1.2.3-alpha1", "2.3.5"]);
}
#[test]
pub fn test_whitespace_delimited_comparator_sets() {
let r = req("> 0.0.9 <= 2.5.3");
assert_eq!(r.to_string(), ">0.0.9, <=2.5.3".to_string());
assert_match(&r, &["0.0.10", "1.0.0", "2.5.3"]);
assert_not_match(&r, &["0.0.8", "2.5.4"]);
}
#[test]
pub fn test_multiple_npm() {
let r = req_npm("> 0.0.9, <= 2.5.3");
assert_eq!(r.to_string(), ">0.0.9 <=2.5.3".to_string());
assert_match(&r, &["0.0.10", "1.0.0", "2.5.3"]);
assert_not_match(&r, &["0.0.8", "2.5.4"]);
let r = req_npm("0.3.0, 0.4.0");
assert_eq!(r.to_string(), "=0.3.0 =0.4.0".to_string());
assert_not_match(&r, &["0.0.8", "0.3.0", "0.4.0"]);
let r = req_npm("<= 0.2.0, >= 0.5.0");
assert_eq!(r.to_string(), "<=0.2.0 >=0.5.0".to_string());
assert_not_match(&r, &["0.0.8", "0.3.0", "0.5.1"]);
let r = req_npm("0.1.0, 0.1.4, 0.1.6");
assert_eq!(r.to_string(), "=0.1.0 =0.1.4 =0.1.6".to_string());
assert_not_match(&r, &["0.1.0", "0.1.4", "0.1.6", "0.2.0"]);
assert!(VersionReq::parse("> 0.1.0,").is_err());
assert!(VersionReq::parse("> 0.3.0, ,").is_err());
let r = req_npm(">=0.5.1-alpha3, <0.6");
assert_eq!(r.to_string(), ">=0.5.1-alpha3 <0.6.0".to_string());
assert_match(
&r,
&[
"0.5.1-alpha3",
"0.5.1-alpha4",
"0.5.1-beta",
"0.5.1",
"0.5.5",
],
);
assert_not_match(
&r,
&["0.5.1-alpha1", "0.5.2-alpha3", "0.5.5-pre", "0.5.0-pre"],
);
assert_not_match(&r, &["0.6.0", "0.6.0-pre"]);
}
#[test]
pub fn test_parsing_tilde() {
let r = req("~1");
assert_match(&r, &["1.0.0", "1.0.1", "1.1.1"]);
assert_not_match(&r, &["0.9.1", "2.9.0", "0.0.9"]);
let r = req("~1.2");
assert_match(&r, &["1.2.0", "1.2.1"]);
assert_not_match(&r, &["1.1.1", "1.3.0", "0.0.9"]);
let r = req("~1.2.2");
assert_match(&r, &["1.2.2", "1.2.4"]);
assert_not_match(&r, &["1.2.1", "1.9.0", "1.0.9", "2.0.1", "0.1.3"]);
let r = req("~1.2.3-beta.2");
assert_match(&r, &["1.2.3", "1.2.4", "1.2.3-beta.2", "1.2.3-beta.4"]);
assert_not_match(&r, &["1.3.3", "1.1.4", "1.2.3-beta.1", "1.2.4-beta.2"]);
}
#[test]
pub fn test_parsing_compatible() {
let r = req("^1");
assert_match(&r, &["1.1.2", "1.1.0", "1.2.1", "1.0.1"]);
assert_not_match(&r, &["0.9.1", "2.9.0", "0.1.4"]);
assert_not_match(&r, &["1.0.0-beta1", "0.1.0-alpha", "1.0.1-pre"]);
let r = req("^1.1");
assert_match(&r, &["1.1.2", "1.1.0", "1.2.1"]);
assert_not_match(&r, &["0.9.1", "2.9.0", "1.0.1", "0.1.4"]);
let r = req("^1.1.2");
assert_match(&r, &["1.1.2", "1.1.4", "1.2.1"]);
assert_not_match(&r, &["0.9.1", "2.9.0", "1.1.1", "0.0.1"]);
assert_not_match(&r, &["1.1.2-alpha1", "1.1.3-alpha1", "2.9.0-alpha1"]);
let r = req("^0.1.2");
assert_match(&r, &["0.1.2", "0.1.4"]);
assert_not_match(&r, &["0.9.1", "2.9.0", "1.1.1", "0.0.1"]);
assert_not_match(&r, &["0.1.2-beta", "0.1.3-alpha", "0.2.0-pre"]);
let r = req("^0.5.1-alpha3");
assert_match(
&r,
&[
"0.5.1-alpha3",
"0.5.1-alpha4",
"0.5.1-beta",
"0.5.1",
"0.5.5",
],
);
assert_not_match(
&r,
&[
"0.5.1-alpha1",
"0.5.2-alpha3",
"0.5.5-pre",
"0.5.0-pre",
"0.6.0",
],
);
let r = req("^0.0.2");
assert_match(&r, &["0.0.2"]);
assert_not_match(&r, &["0.9.1", "2.9.0", "1.1.1", "0.0.1", "0.1.4"]);
let r = req("^0.0");
assert_match(&r, &["0.0.2", "0.0.0"]);
assert_not_match(&r, &["0.9.1", "2.9.0", "1.1.1", "0.1.4"]);
let r = req("^0");
assert_match(&r, &["0.9.1", "0.0.2", "0.0.0"]);
assert_not_match(&r, &["2.9.0", "1.1.1"]);
let r = req("^1.4.2-beta.5");
assert_match(
&r,
&["1.4.2", "1.4.3", "1.4.2-beta.5", "1.4.2-beta.6", "1.4.2-c"],
);
assert_not_match(
&r,
&[
"0.9.9",
"2.0.0",
"1.4.2-alpha",
"1.4.2-beta.4",
"1.4.3-beta.5",
],
);
}
#[test]
pub fn test_parsing_wildcard() {
let r = req("");
assert_match(&r, &["0.9.1", "2.9.0", "0.0.9", "1.0.1", "1.1.1"]);
assert_not_match(&r, &[]);
let r = req("*");
assert_match(&r, &["0.9.1", "2.9.0", "0.0.9", "1.0.1", "1.1.1"]);
assert_not_match(&r, &[]);
let r = req("x");
assert_match(&r, &["0.9.1", "2.9.0", "0.0.9", "1.0.1", "1.1.1"]);
assert_not_match(&r, &[]);
let r = req("X");
assert_match(&r, &["0.9.1", "2.9.0", "0.0.9", "1.0.1", "1.1.1"]);
assert_not_match(&r, &[]);
let r = req("1.*");
assert_match(&r, &["1.2.0", "1.2.1", "1.1.1", "1.3.0"]);
assert_not_match(&r, &["0.0.9"]);
let r = req("1.x");
assert_match(&r, &["1.2.0", "1.2.1", "1.1.1", "1.3.0"]);
assert_not_match(&r, &["0.0.9"]);
let r = req("1.X");
assert_match(&r, &["1.2.0", "1.2.1", "1.1.1", "1.3.0"]);
assert_not_match(&r, &["0.0.9"]);
let r = req("1.2.*");
assert_match(&r, &["1.2.0", "1.2.2", "1.2.4"]);
assert_not_match(&r, &["1.9.0", "1.0.9", "2.0.1", "0.1.3"]);
let r = req("1.2.x");
assert_match(&r, &["1.2.0", "1.2.2", "1.2.4"]);
assert_not_match(&r, &["1.9.0", "1.0.9", "2.0.1", "0.1.3"]);
let r = req("1.2.X");
assert_match(&r, &["1.2.0", "1.2.2", "1.2.4"]);
assert_not_match(&r, &["1.9.0", "1.0.9", "2.0.1", "0.1.3"]);
}
#[test]
pub fn test_parsing_logical_or() {
let r = req("=1.2.3 || =2.3.4");
assert_eq!(r.to_string(), "=1.2.3 || =2.3.4".to_string());
assert_match(&r, &["1.2.3", "2.3.4"]);
assert_not_match(&r, &["1.0.0", "2.9.0", "0.1.4"]);
assert_not_match(&r, &["1.2.3-beta1", "2.3.4-alpha", "1.2.3-pre"]);
let r = req("1.1 || =1.2.3");
assert_eq!(r.to_string(), ">=1.1.0, <1.2.0 || =1.2.3".to_string());
assert_match(&r, &["1.1.0", "1.1.12", "1.2.3"]);
assert_not_match(&r, &["1.0.0", "1.2.2", "1.3.0"]);
let r = req("6.* || 8.* || >= 10.*");
assert_eq!(
r.to_string(),
">=6.0.0, <7.0.0 || >=8.0.0, <9.0.0 || >=10.0.0".to_string()
);
assert_match(&r, &["6.0.0", "6.1.2"]);
assert_match(&r, &["8.0.0", "8.2.4"]);
assert_match(&r, &["10.1.2", "11.3.4"]);
assert_not_match(&r, &["5.0.0", "7.0.0", "9.0.0"]);
}
#[test]
pub fn test_parsing_logical_or_npm() {
let r = req_npm("=1.2.3 || =2.3.4");
assert_eq!(r.to_string(), "=1.2.3 || =2.3.4".to_string());
assert_match(&r, &["1.2.3", "2.3.4"]);
assert_not_match(&r, &["1.0.0", "2.9.0", "0.1.4"]);
assert_not_match(&r, &["1.2.3-beta1", "2.3.4-alpha", "1.2.3-pre"]);
let r = req_npm("1.1 || =1.2.3");
assert_eq!(r.to_string(), ">=1.1.0 <1.2.0 || =1.2.3".to_string());
assert_match(&r, &["1.1.0", "1.1.12", "1.2.3"]);
assert_not_match(&r, &["1.0.0", "1.2.2", "1.3.0"]);
let r = req_npm("6.* || 8.* || >= 10.*");
assert_eq!(
r.to_string(),
">=6.0.0 <7.0.0 || >=8.0.0 <9.0.0 || >=10.0.0".to_string()
);
assert_match(&r, &["6.0.0", "6.1.2"]);
assert_match(&r, &["8.0.0", "8.2.4"]);
assert_match(&r, &["10.1.2", "11.3.4"]);
assert_not_match(&r, &["5.0.0", "7.0.0", "9.0.0"]);
}
#[test]
pub fn test_any() {
let r = VersionReq::any();
assert_match(&r, &["0.0.1", "0.1.0", "1.0.0"]);
}
#[test]
pub fn test_pre() {
let r = req("=2.1.1-really.0");
assert_match(&r, &["2.1.1-really.0"]);
}
#[test]
pub fn test_from_str() {
assert_eq!(
"1.0.0".parse::<VersionReq>().unwrap().to_string(),
">=1.0.0, <2.0.0".to_string()
);
assert_eq!(
"=1.0.0".parse::<VersionReq>().unwrap().to_string(),
"=1.0.0".to_string()
);
assert_eq!(
"~1".parse::<VersionReq>().unwrap().to_string(),
">=1.0.0, <2.0.0".to_string()
);
assert_eq!(
"~1.2".parse::<VersionReq>().unwrap().to_string(),
">=1.2.0, <1.3.0".to_string()
);
assert_eq!(
"^1".parse::<VersionReq>().unwrap().to_string(),
">=1.0.0, <2.0.0".to_string()
);
assert_eq!(
"^1.1".parse::<VersionReq>().unwrap().to_string(),
">=1.1.0, <2.0.0".to_string()
);
assert_eq!(
"*".parse::<VersionReq>().unwrap().to_string(),
">=0.0.0".to_string()
);
assert_eq!(
"1.*".parse::<VersionReq>().unwrap().to_string(),
">=1.0.0, <2.0.0".to_string()
);
assert_eq!(
"< 1.0.0".parse::<VersionReq>().unwrap().to_string(),
"<1.0.0".to_string()
);
}
#[test]
fn test_cargo3202() {
let v = "0.*.*".parse::<VersionReq>().unwrap();
assert_eq!(">=0.0.0, <1.0.0", format!("{}", v.ranges[0]));
let v = "0.0.*".parse::<VersionReq>().unwrap();
assert_eq!(">=0.0.0, <0.1.0", format!("{}", v.ranges[0]));
let r = req("0.*.*");
assert_match(&r, &["0.5.0"]);
}
#[test]
fn test_eq_hash() {
assert!(req("^1") == req("^1"));
assert!(calculate_hash(req("^1")) == calculate_hash(req("^1")));
assert!(req("^1") != req("^2"));
}
#[test]
fn test_ordering() {
assert!(req("=1") > req("*"));
assert!(req(">1") < req("*"));
assert!(req(">=1") > req("*"));
assert!(req("<1") > req("*"));
assert!(req("<=1") > req("*"));
assert!(req("~1") > req("*"));
assert!(req("^1") > req("*"));
assert!(req("*") == req("*"));
}
#[test]
fn is_exact() {
assert!(req("=1.0.0").is_exact());
assert!(req("=1.0.0-alpha").is_exact());
assert!(!req("=1").is_exact());
assert!(!req(">=1.0.0").is_exact());
assert!(!req(">=1.0.0, <2.0.0").is_exact());
}
}