| use crate::{Comparator, Op, Version, VersionReq}; | 
 |  | 
 | pub(crate) fn matches_req(req: &VersionReq, ver: &Version) -> bool { | 
 |     for cmp in &req.comparators { | 
 |         if !matches_impl(cmp, ver) { | 
 |             return false; | 
 |         } | 
 |     } | 
 |  | 
 |     if ver.pre.is_empty() { | 
 |         return true; | 
 |     } | 
 |  | 
 |     // If a version has a prerelease tag (for example, 1.2.3-alpha.3) then it | 
 |     // will only be allowed to satisfy req if at least one comparator with the | 
 |     // same major.minor.patch also has a prerelease tag. | 
 |     for cmp in &req.comparators { | 
 |         if pre_is_compatible(cmp, ver) { | 
 |             return true; | 
 |         } | 
 |     } | 
 |  | 
 |     false | 
 | } | 
 |  | 
 | pub(crate) fn matches_comparator(cmp: &Comparator, ver: &Version) -> bool { | 
 |     matches_impl(cmp, ver) && (ver.pre.is_empty() || pre_is_compatible(cmp, ver)) | 
 | } | 
 |  | 
 | fn matches_impl(cmp: &Comparator, ver: &Version) -> bool { | 
 |     match cmp.op { | 
 |         Op::Exact | Op::Wildcard => matches_exact(cmp, ver), | 
 |         Op::Greater => matches_greater(cmp, ver), | 
 |         Op::GreaterEq => matches_exact(cmp, ver) || matches_greater(cmp, ver), | 
 |         Op::Less => matches_less(cmp, ver), | 
 |         Op::LessEq => matches_exact(cmp, ver) || matches_less(cmp, ver), | 
 |         Op::Tilde => matches_tilde(cmp, ver), | 
 |         Op::Caret => matches_caret(cmp, ver), | 
 |         #[cfg(no_non_exhaustive)] | 
 |         Op::__NonExhaustive => unreachable!(), | 
 |     } | 
 | } | 
 |  | 
 | fn matches_exact(cmp: &Comparator, ver: &Version) -> bool { | 
 |     if ver.major != cmp.major { | 
 |         return false; | 
 |     } | 
 |  | 
 |     if let Some(minor) = cmp.minor { | 
 |         if ver.minor != minor { | 
 |             return false; | 
 |         } | 
 |     } | 
 |  | 
 |     if let Some(patch) = cmp.patch { | 
 |         if ver.patch != patch { | 
 |             return false; | 
 |         } | 
 |     } | 
 |  | 
 |     ver.pre == cmp.pre | 
 | } | 
 |  | 
 | fn matches_greater(cmp: &Comparator, ver: &Version) -> bool { | 
 |     if ver.major != cmp.major { | 
 |         return ver.major > cmp.major; | 
 |     } | 
 |  | 
 |     match cmp.minor { | 
 |         None => return false, | 
 |         Some(minor) => { | 
 |             if ver.minor != minor { | 
 |                 return ver.minor > minor; | 
 |             } | 
 |         } | 
 |     } | 
 |  | 
 |     match cmp.patch { | 
 |         None => return false, | 
 |         Some(patch) => { | 
 |             if ver.patch != patch { | 
 |                 return ver.patch > patch; | 
 |             } | 
 |         } | 
 |     } | 
 |  | 
 |     ver.pre > cmp.pre | 
 | } | 
 |  | 
 | fn matches_less(cmp: &Comparator, ver: &Version) -> bool { | 
 |     if ver.major != cmp.major { | 
 |         return ver.major < cmp.major; | 
 |     } | 
 |  | 
 |     match cmp.minor { | 
 |         None => return false, | 
 |         Some(minor) => { | 
 |             if ver.minor != minor { | 
 |                 return ver.minor < minor; | 
 |             } | 
 |         } | 
 |     } | 
 |  | 
 |     match cmp.patch { | 
 |         None => return false, | 
 |         Some(patch) => { | 
 |             if ver.patch != patch { | 
 |                 return ver.patch < patch; | 
 |             } | 
 |         } | 
 |     } | 
 |  | 
 |     ver.pre < cmp.pre | 
 | } | 
 |  | 
 | fn matches_tilde(cmp: &Comparator, ver: &Version) -> bool { | 
 |     if ver.major != cmp.major { | 
 |         return false; | 
 |     } | 
 |  | 
 |     if let Some(minor) = cmp.minor { | 
 |         if ver.minor != minor { | 
 |             return false; | 
 |         } | 
 |     } | 
 |  | 
 |     if let Some(patch) = cmp.patch { | 
 |         if ver.patch != patch { | 
 |             return ver.patch > patch; | 
 |         } | 
 |     } | 
 |  | 
 |     ver.pre >= cmp.pre | 
 | } | 
 |  | 
 | fn matches_caret(cmp: &Comparator, ver: &Version) -> bool { | 
 |     if ver.major != cmp.major { | 
 |         return false; | 
 |     } | 
 |  | 
 |     let minor = match cmp.minor { | 
 |         None => return true, | 
 |         Some(minor) => minor, | 
 |     }; | 
 |  | 
 |     let patch = match cmp.patch { | 
 |         None => { | 
 |             if cmp.major > 0 { | 
 |                 return ver.minor >= minor; | 
 |             } else { | 
 |                 return ver.minor == minor; | 
 |             } | 
 |         } | 
 |         Some(patch) => patch, | 
 |     }; | 
 |  | 
 |     if cmp.major > 0 { | 
 |         if ver.minor != minor { | 
 |             return ver.minor > minor; | 
 |         } else if ver.patch != patch { | 
 |             return ver.patch > patch; | 
 |         } | 
 |     } else if minor > 0 { | 
 |         if ver.minor != minor { | 
 |             return false; | 
 |         } else if ver.patch != patch { | 
 |             return ver.patch > patch; | 
 |         } | 
 |     } else if ver.minor != minor || ver.patch != patch { | 
 |         return false; | 
 |     } | 
 |  | 
 |     ver.pre >= cmp.pre | 
 | } | 
 |  | 
 | fn pre_is_compatible(cmp: &Comparator, ver: &Version) -> bool { | 
 |     cmp.major == ver.major | 
 |         && cmp.minor == Some(ver.minor) | 
 |         && cmp.patch == Some(ver.patch) | 
 |         && !cmp.pre.is_empty() | 
 | } |