16acc7838Sopenharmony_ci//! This tiny crate checks that the running or installed `rustc` meets some
26acc7838Sopenharmony_ci//! version requirements. The version is queried by calling the Rust compiler
36acc7838Sopenharmony_ci//! with `--version`. The path to the compiler is determined first via the
46acc7838Sopenharmony_ci//! `RUSTC` environment variable. If it is not set, then `rustc` is used. If
56acc7838Sopenharmony_ci//! that fails, no determination is made, and calls return `None`.
66acc7838Sopenharmony_ci//!
76acc7838Sopenharmony_ci//! # Examples
86acc7838Sopenharmony_ci//!
96acc7838Sopenharmony_ci//! * Set a `cfg` flag in `build.rs` if the running compiler was determined to
106acc7838Sopenharmony_ci//!   be at least version `1.13.0`:
116acc7838Sopenharmony_ci//!
126acc7838Sopenharmony_ci//!   ```rust
136acc7838Sopenharmony_ci//!   extern crate version_check as rustc;
146acc7838Sopenharmony_ci//!
156acc7838Sopenharmony_ci//!   if rustc::is_min_version("1.13.0").unwrap_or(false) {
166acc7838Sopenharmony_ci//!       println!("cargo:rustc-cfg=question_mark_operator");
176acc7838Sopenharmony_ci//!   }
186acc7838Sopenharmony_ci//!   ```
196acc7838Sopenharmony_ci//!
206acc7838Sopenharmony_ci//!   See [`is_max_version`] or [`is_exact_version`] to check if the compiler
216acc7838Sopenharmony_ci//!   is _at most_ or _exactly_ a certain version.
226acc7838Sopenharmony_ci//!
236acc7838Sopenharmony_ci//! * Check that the running compiler was released on or after `2018-12-18`:
246acc7838Sopenharmony_ci//!
256acc7838Sopenharmony_ci//!   ```rust
266acc7838Sopenharmony_ci//!   extern crate version_check as rustc;
276acc7838Sopenharmony_ci//!
286acc7838Sopenharmony_ci//!   match rustc::is_min_date("2018-12-18") {
296acc7838Sopenharmony_ci//!       Some(true) => "Yep! It's recent!",
306acc7838Sopenharmony_ci//!       Some(false) => "No, it's older.",
316acc7838Sopenharmony_ci//!       None => "Couldn't determine the rustc version."
326acc7838Sopenharmony_ci//!   };
336acc7838Sopenharmony_ci//!   ```
346acc7838Sopenharmony_ci//!
356acc7838Sopenharmony_ci//!   See [`is_max_date`] or [`is_exact_date`] to check if the compiler was
366acc7838Sopenharmony_ci//!   released _prior to_ or _exactly on_ a certain date.
376acc7838Sopenharmony_ci//!
386acc7838Sopenharmony_ci//! * Check that the running compiler supports feature flags:
396acc7838Sopenharmony_ci//!
406acc7838Sopenharmony_ci//!   ```rust
416acc7838Sopenharmony_ci//!   extern crate version_check as rustc;
426acc7838Sopenharmony_ci//!
436acc7838Sopenharmony_ci//!   match rustc::is_feature_flaggable() {
446acc7838Sopenharmony_ci//!       Some(true) => "Yes! It's a dev or nightly release!",
456acc7838Sopenharmony_ci//!       Some(false) => "No, it's stable or beta.",
466acc7838Sopenharmony_ci//!       None => "Couldn't determine the rustc version."
476acc7838Sopenharmony_ci//!   };
486acc7838Sopenharmony_ci//!   ```
496acc7838Sopenharmony_ci//!
506acc7838Sopenharmony_ci//! * Check that the running compiler supports a specific feature:
516acc7838Sopenharmony_ci//!
526acc7838Sopenharmony_ci//!   ```rust
536acc7838Sopenharmony_ci//!   extern crate version_check as rustc;
546acc7838Sopenharmony_ci//!
556acc7838Sopenharmony_ci//!   if let Some(true) = rustc::supports_feature("doc_cfg") {
566acc7838Sopenharmony_ci//!      println!("cargo:rustc-cfg=has_doc_cfg");
576acc7838Sopenharmony_ci//!   }
586acc7838Sopenharmony_ci//!   ```
596acc7838Sopenharmony_ci//!
606acc7838Sopenharmony_ci//! * Check that the running compiler is on the stable channel:
616acc7838Sopenharmony_ci//!
626acc7838Sopenharmony_ci//!   ```rust
636acc7838Sopenharmony_ci//!   extern crate version_check as rustc;
646acc7838Sopenharmony_ci//!
656acc7838Sopenharmony_ci//!   match rustc::Channel::read() {
666acc7838Sopenharmony_ci//!       Some(c) if c.is_stable() => format!("Yes! It's stable."),
676acc7838Sopenharmony_ci//!       Some(c) => format!("No, the channel {} is not stable.", c),
686acc7838Sopenharmony_ci//!       None => format!("Couldn't determine the rustc version.")
696acc7838Sopenharmony_ci//!   };
706acc7838Sopenharmony_ci//!   ```
716acc7838Sopenharmony_ci//!
726acc7838Sopenharmony_ci//! To interact with the version, release date, and release channel as structs,
736acc7838Sopenharmony_ci//! use [`Version`], [`Date`], and [`Channel`], respectively. The [`triple()`]
746acc7838Sopenharmony_ci//! function returns all three values efficiently.
756acc7838Sopenharmony_ci//!
766acc7838Sopenharmony_ci//! # Alternatives
776acc7838Sopenharmony_ci//!
786acc7838Sopenharmony_ci//! This crate is dead simple with no dependencies. If you need something more
796acc7838Sopenharmony_ci//! and don't care about panicking if the version cannot be obtained, or if you
806acc7838Sopenharmony_ci//! don't mind adding dependencies, see
816acc7838Sopenharmony_ci//! [rustc_version](https://crates.io/crates/rustc_version).
826acc7838Sopenharmony_ci
836acc7838Sopenharmony_ci#![allow(deprecated)]
846acc7838Sopenharmony_ci
856acc7838Sopenharmony_cimod version;
866acc7838Sopenharmony_cimod channel;
876acc7838Sopenharmony_cimod date;
886acc7838Sopenharmony_ci
896acc7838Sopenharmony_ciuse std::env;
906acc7838Sopenharmony_ciuse std::process::Command;
916acc7838Sopenharmony_ci
926acc7838Sopenharmony_ci#[doc(inline)] pub use version::*;
936acc7838Sopenharmony_ci#[doc(inline)] pub use channel::*;
946acc7838Sopenharmony_ci#[doc(inline)] pub use date::*;
956acc7838Sopenharmony_ci
966acc7838Sopenharmony_ci/// Parses (version, date) as available from rustc version string.
976acc7838Sopenharmony_cifn version_and_date_from_rustc_version(s: &str) -> (Option<String>, Option<String>) {
986acc7838Sopenharmony_ci    let last_line = s.lines().last().unwrap_or(s);
996acc7838Sopenharmony_ci    let mut components = last_line.trim().split(" ");
1006acc7838Sopenharmony_ci    let version = components.nth(1);
1016acc7838Sopenharmony_ci    let date = components.filter(|c| c.ends_with(')')).next()
1026acc7838Sopenharmony_ci        .map(|s| s.trim_right().trim_right_matches(")").trim_left().trim_left_matches('('));
1036acc7838Sopenharmony_ci    (version.map(|s| s.to_string()), date.map(|s| s.to_string()))
1046acc7838Sopenharmony_ci}
1056acc7838Sopenharmony_ci
1066acc7838Sopenharmony_ci/// Parses (version, date) as available from rustc verbose version output.
1076acc7838Sopenharmony_cifn version_and_date_from_rustc_verbose_version(s: &str) -> (Option<String>, Option<String>) {
1086acc7838Sopenharmony_ci    let (mut version, mut date) = (None, None);
1096acc7838Sopenharmony_ci    for line in s.lines() {
1106acc7838Sopenharmony_ci        let split = |s: &str| s.splitn(2, ":").nth(1).map(|s| s.trim().to_string());
1116acc7838Sopenharmony_ci        match line.trim().split(" ").nth(0) {
1126acc7838Sopenharmony_ci            Some("rustc") => {
1136acc7838Sopenharmony_ci                let (v, d) = version_and_date_from_rustc_version(line);
1146acc7838Sopenharmony_ci                version = version.or(v);
1156acc7838Sopenharmony_ci                date = date.or(d);
1166acc7838Sopenharmony_ci            },
1176acc7838Sopenharmony_ci            Some("release:") => version = split(line),
1186acc7838Sopenharmony_ci            Some("commit-date:") if line.ends_with("unknown") => date = None,
1196acc7838Sopenharmony_ci            Some("commit-date:") => date = split(line),
1206acc7838Sopenharmony_ci            _ => continue
1216acc7838Sopenharmony_ci        }
1226acc7838Sopenharmony_ci    }
1236acc7838Sopenharmony_ci
1246acc7838Sopenharmony_ci    (version, date)
1256acc7838Sopenharmony_ci}
1266acc7838Sopenharmony_ci
1276acc7838Sopenharmony_ci/// Returns (version, date) as available from `rustc --version`.
1286acc7838Sopenharmony_cifn get_version_and_date() -> Option<(Option<String>, Option<String>)> {
1296acc7838Sopenharmony_ci    let rustc = env::var("RUSTC").unwrap_or_else(|_| "rustc".to_string());
1306acc7838Sopenharmony_ci    Command::new(rustc).arg("--verbose").arg("--version").output().ok()
1316acc7838Sopenharmony_ci        .and_then(|output| String::from_utf8(output.stdout).ok())
1326acc7838Sopenharmony_ci        .map(|s| version_and_date_from_rustc_verbose_version(&s))
1336acc7838Sopenharmony_ci}
1346acc7838Sopenharmony_ci
1356acc7838Sopenharmony_ci/// Reads the triple of [`Version`], [`Channel`], and [`Date`] of the installed
1366acc7838Sopenharmony_ci/// or running `rustc`.
1376acc7838Sopenharmony_ci///
1386acc7838Sopenharmony_ci/// If any attribute cannot be determined (see the [top-level
1396acc7838Sopenharmony_ci/// documentation](crate)), returns `None`.
1406acc7838Sopenharmony_ci///
1416acc7838Sopenharmony_ci/// To obtain only one of three attributes, use [`Version::read()`],
1426acc7838Sopenharmony_ci/// [`Channel::read()`], or [`Date::read()`].
1436acc7838Sopenharmony_cipub fn triple() -> Option<(Version, Channel, Date)> {
1446acc7838Sopenharmony_ci    let (version_str, date_str) = match get_version_and_date() {
1456acc7838Sopenharmony_ci        Some((Some(version), Some(date))) => (version, date),
1466acc7838Sopenharmony_ci        _ => return None
1476acc7838Sopenharmony_ci    };
1486acc7838Sopenharmony_ci
1496acc7838Sopenharmony_ci    // Can't use `?` or `try!` for `Option` in 1.0.0.
1506acc7838Sopenharmony_ci    match Version::parse(&version_str) {
1516acc7838Sopenharmony_ci        Some(version) => match Channel::parse(&version_str) {
1526acc7838Sopenharmony_ci            Some(channel) => match Date::parse(&date_str) {
1536acc7838Sopenharmony_ci                Some(date) => Some((version, channel, date)),
1546acc7838Sopenharmony_ci                _ => None,
1556acc7838Sopenharmony_ci            },
1566acc7838Sopenharmony_ci            _ => None,
1576acc7838Sopenharmony_ci        },
1586acc7838Sopenharmony_ci        _ => None
1596acc7838Sopenharmony_ci    }
1606acc7838Sopenharmony_ci}
1616acc7838Sopenharmony_ci
1626acc7838Sopenharmony_ci/// Checks that the running or installed `rustc` was released **on or after**
1636acc7838Sopenharmony_ci/// some date.
1646acc7838Sopenharmony_ci///
1656acc7838Sopenharmony_ci/// The format of `min_date` must be YYYY-MM-DD. For instance: `2016-12-20` or
1666acc7838Sopenharmony_ci/// `2017-01-09`.
1676acc7838Sopenharmony_ci///
1686acc7838Sopenharmony_ci/// If the date cannot be retrieved or parsed, or if `min_date` could not be
1696acc7838Sopenharmony_ci/// parsed, returns `None`. Otherwise returns `true` if the installed `rustc`
1706acc7838Sopenharmony_ci/// was release on or after `min_date` and `false` otherwise.
1716acc7838Sopenharmony_cipub fn is_min_date(min_date: &str) -> Option<bool> {
1726acc7838Sopenharmony_ci    match (Date::read(), Date::parse(min_date)) {
1736acc7838Sopenharmony_ci        (Some(rustc_date), Some(min_date)) => Some(rustc_date >= min_date),
1746acc7838Sopenharmony_ci        _ => None
1756acc7838Sopenharmony_ci    }
1766acc7838Sopenharmony_ci}
1776acc7838Sopenharmony_ci
1786acc7838Sopenharmony_ci/// Checks that the running or installed `rustc` was released **on or before**
1796acc7838Sopenharmony_ci/// some date.
1806acc7838Sopenharmony_ci///
1816acc7838Sopenharmony_ci/// The format of `max_date` must be YYYY-MM-DD. For instance: `2016-12-20` or
1826acc7838Sopenharmony_ci/// `2017-01-09`.
1836acc7838Sopenharmony_ci///
1846acc7838Sopenharmony_ci/// If the date cannot be retrieved or parsed, or if `max_date` could not be
1856acc7838Sopenharmony_ci/// parsed, returns `None`. Otherwise returns `true` if the installed `rustc`
1866acc7838Sopenharmony_ci/// was release on or before `max_date` and `false` otherwise.
1876acc7838Sopenharmony_cipub fn is_max_date(max_date: &str) -> Option<bool> {
1886acc7838Sopenharmony_ci    match (Date::read(), Date::parse(max_date)) {
1896acc7838Sopenharmony_ci        (Some(rustc_date), Some(max_date)) => Some(rustc_date <= max_date),
1906acc7838Sopenharmony_ci        _ => None
1916acc7838Sopenharmony_ci    }
1926acc7838Sopenharmony_ci}
1936acc7838Sopenharmony_ci
1946acc7838Sopenharmony_ci/// Checks that the running or installed `rustc` was released **exactly** on
1956acc7838Sopenharmony_ci/// some date.
1966acc7838Sopenharmony_ci///
1976acc7838Sopenharmony_ci/// The format of `date` must be YYYY-MM-DD. For instance: `2016-12-20` or
1986acc7838Sopenharmony_ci/// `2017-01-09`.
1996acc7838Sopenharmony_ci///
2006acc7838Sopenharmony_ci/// If the date cannot be retrieved or parsed, or if `date` could not be parsed,
2016acc7838Sopenharmony_ci/// returns `None`. Otherwise returns `true` if the installed `rustc` was
2026acc7838Sopenharmony_ci/// release on `date` and `false` otherwise.
2036acc7838Sopenharmony_cipub fn is_exact_date(date: &str) -> Option<bool> {
2046acc7838Sopenharmony_ci    match (Date::read(), Date::parse(date)) {
2056acc7838Sopenharmony_ci        (Some(rustc_date), Some(date)) => Some(rustc_date == date),
2066acc7838Sopenharmony_ci        _ => None
2076acc7838Sopenharmony_ci    }
2086acc7838Sopenharmony_ci}
2096acc7838Sopenharmony_ci
2106acc7838Sopenharmony_ci/// Checks that the running or installed `rustc` is **at least** some minimum
2116acc7838Sopenharmony_ci/// version.
2126acc7838Sopenharmony_ci///
2136acc7838Sopenharmony_ci/// The format of `min_version` is a semantic version: `1.3.0`, `1.15.0-beta`,
2146acc7838Sopenharmony_ci/// `1.14.0`, `1.16.0-nightly`, etc.
2156acc7838Sopenharmony_ci///
2166acc7838Sopenharmony_ci/// If the version cannot be retrieved or parsed, or if `min_version` could not
2176acc7838Sopenharmony_ci/// be parsed, returns `None`. Otherwise returns `true` if the installed `rustc`
2186acc7838Sopenharmony_ci/// is at least `min_version` and `false` otherwise.
2196acc7838Sopenharmony_cipub fn is_min_version(min_version: &str) -> Option<bool> {
2206acc7838Sopenharmony_ci    match (Version::read(), Version::parse(min_version)) {
2216acc7838Sopenharmony_ci        (Some(rustc_ver), Some(min_ver)) => Some(rustc_ver >= min_ver),
2226acc7838Sopenharmony_ci        _ => None
2236acc7838Sopenharmony_ci    }
2246acc7838Sopenharmony_ci}
2256acc7838Sopenharmony_ci
2266acc7838Sopenharmony_ci/// Checks that the running or installed `rustc` is **at most** some maximum
2276acc7838Sopenharmony_ci/// version.
2286acc7838Sopenharmony_ci///
2296acc7838Sopenharmony_ci/// The format of `max_version` is a semantic version: `1.3.0`, `1.15.0-beta`,
2306acc7838Sopenharmony_ci/// `1.14.0`, `1.16.0-nightly`, etc.
2316acc7838Sopenharmony_ci///
2326acc7838Sopenharmony_ci/// If the version cannot be retrieved or parsed, or if `max_version` could not
2336acc7838Sopenharmony_ci/// be parsed, returns `None`. Otherwise returns `true` if the installed `rustc`
2346acc7838Sopenharmony_ci/// is at most `max_version` and `false` otherwise.
2356acc7838Sopenharmony_cipub fn is_max_version(max_version: &str) -> Option<bool> {
2366acc7838Sopenharmony_ci    match (Version::read(), Version::parse(max_version)) {
2376acc7838Sopenharmony_ci        (Some(rustc_ver), Some(max_ver)) => Some(rustc_ver <= max_ver),
2386acc7838Sopenharmony_ci        _ => None
2396acc7838Sopenharmony_ci    }
2406acc7838Sopenharmony_ci}
2416acc7838Sopenharmony_ci
2426acc7838Sopenharmony_ci/// Checks that the running or installed `rustc` is **exactly** some version.
2436acc7838Sopenharmony_ci///
2446acc7838Sopenharmony_ci/// The format of `version` is a semantic version: `1.3.0`, `1.15.0-beta`,
2456acc7838Sopenharmony_ci/// `1.14.0`, `1.16.0-nightly`, etc.
2466acc7838Sopenharmony_ci///
2476acc7838Sopenharmony_ci/// If the version cannot be retrieved or parsed, or if `version` could not be
2486acc7838Sopenharmony_ci/// parsed, returns `None`. Otherwise returns `true` if the installed `rustc` is
2496acc7838Sopenharmony_ci/// exactly `version` and `false` otherwise.
2506acc7838Sopenharmony_cipub fn is_exact_version(version: &str) -> Option<bool> {
2516acc7838Sopenharmony_ci    match (Version::read(), Version::parse(version)) {
2526acc7838Sopenharmony_ci        (Some(rustc_ver), Some(version)) => Some(rustc_ver == version),
2536acc7838Sopenharmony_ci        _ => None
2546acc7838Sopenharmony_ci    }
2556acc7838Sopenharmony_ci}
2566acc7838Sopenharmony_ci
2576acc7838Sopenharmony_ci/// Checks whether the running or installed `rustc` supports feature flags.
2586acc7838Sopenharmony_ci///
2596acc7838Sopenharmony_ci/// In other words, if the channel is either "nightly" or "dev".
2606acc7838Sopenharmony_ci///
2616acc7838Sopenharmony_ci/// Note that support for specific `rustc` features can be enabled or disabled
2626acc7838Sopenharmony_ci/// via the `allow-features` compiler flag, which this function _does not_
2636acc7838Sopenharmony_ci/// check. That is, this function _does not_ check whether a _specific_ feature
2646acc7838Sopenharmony_ci/// is supported, but instead whether features are supported at all. To check
2656acc7838Sopenharmony_ci/// for support for a specific feature, use [`supports_feature()`].
2666acc7838Sopenharmony_ci///
2676acc7838Sopenharmony_ci/// If the version could not be determined, returns `None`. Otherwise returns
2686acc7838Sopenharmony_ci/// `true` if the running version supports feature flags and `false` otherwise.
2696acc7838Sopenharmony_cipub fn is_feature_flaggable() -> Option<bool> {
2706acc7838Sopenharmony_ci    Channel::read().map(|c| c.supports_features())
2716acc7838Sopenharmony_ci}
2726acc7838Sopenharmony_ci
2736acc7838Sopenharmony_ci/// Checks whether the running or installed `rustc` supports `feature`.
2746acc7838Sopenharmony_ci///
2756acc7838Sopenharmony_ci/// Returns _true_ _iff_ [`is_feature_flaggable()`] returns `true` _and_ the
2766acc7838Sopenharmony_ci/// feature is not disabled via exclusion in `allow-features` via `RUSTFLAGS` or
2776acc7838Sopenharmony_ci/// `CARGO_ENCODED_RUSTFLAGS`. If the version could not be determined, returns
2786acc7838Sopenharmony_ci/// `None`.
2796acc7838Sopenharmony_ci///
2806acc7838Sopenharmony_ci/// # Example
2816acc7838Sopenharmony_ci///
2826acc7838Sopenharmony_ci/// ```rust
2836acc7838Sopenharmony_ci/// use version_check as rustc;
2846acc7838Sopenharmony_ci///
2856acc7838Sopenharmony_ci/// if let Some(true) = rustc::supports_feature("doc_cfg") {
2866acc7838Sopenharmony_ci///    println!("cargo:rustc-cfg=has_doc_cfg");
2876acc7838Sopenharmony_ci/// }
2886acc7838Sopenharmony_ci/// ```
2896acc7838Sopenharmony_cipub fn supports_feature(feature: &str) -> Option<bool> {
2906acc7838Sopenharmony_ci    match is_feature_flaggable() {
2916acc7838Sopenharmony_ci        Some(true) => { /* continue */ }
2926acc7838Sopenharmony_ci        Some(false) => return Some(false),
2936acc7838Sopenharmony_ci        None => return None,
2946acc7838Sopenharmony_ci    }
2956acc7838Sopenharmony_ci
2966acc7838Sopenharmony_ci    let env_flags = env::var_os("CARGO_ENCODED_RUSTFLAGS")
2976acc7838Sopenharmony_ci        .map(|flags| (flags, '\x1f'))
2986acc7838Sopenharmony_ci        .or_else(|| env::var_os("RUSTFLAGS").map(|flags| (flags, ' ')));
2996acc7838Sopenharmony_ci
3006acc7838Sopenharmony_ci    if let Some((flags, delim)) = env_flags {
3016acc7838Sopenharmony_ci        const ALLOW_FEATURES: &'static str = "allow-features=";
3026acc7838Sopenharmony_ci
3036acc7838Sopenharmony_ci        let rustflags = flags.to_string_lossy();
3046acc7838Sopenharmony_ci        let allow_features = rustflags.split(delim)
3056acc7838Sopenharmony_ci            .map(|flag| flag.trim_left_matches("-Z").trim())
3066acc7838Sopenharmony_ci            .filter(|flag| flag.starts_with(ALLOW_FEATURES))
3076acc7838Sopenharmony_ci            .map(|flag| &flag[ALLOW_FEATURES.len()..]);
3086acc7838Sopenharmony_ci
3096acc7838Sopenharmony_ci        if let Some(allow_features) = allow_features.last() {
3106acc7838Sopenharmony_ci            return Some(allow_features.split(',').any(|f| f.trim() == feature));
3116acc7838Sopenharmony_ci        }
3126acc7838Sopenharmony_ci    }
3136acc7838Sopenharmony_ci
3146acc7838Sopenharmony_ci    // If there are no `RUSTFLAGS` or `CARGO_ENCODED_RUSTFLAGS` or they don't
3156acc7838Sopenharmony_ci    // contain an `allow-features` flag, assume compiler allows all features.
3166acc7838Sopenharmony_ci    Some(true)
3176acc7838Sopenharmony_ci}
3186acc7838Sopenharmony_ci
3196acc7838Sopenharmony_ci#[cfg(test)]
3206acc7838Sopenharmony_cimod tests {
3216acc7838Sopenharmony_ci    use std::{env, fs};
3226acc7838Sopenharmony_ci
3236acc7838Sopenharmony_ci    use super::version_and_date_from_rustc_version;
3246acc7838Sopenharmony_ci    use super::version_and_date_from_rustc_verbose_version;
3256acc7838Sopenharmony_ci
3266acc7838Sopenharmony_ci    macro_rules! check_parse {
3276acc7838Sopenharmony_ci        (@ $f:expr, $s:expr => $v:expr, $d:expr) => ({
3286acc7838Sopenharmony_ci            if let (Some(v), d) = $f(&$s) {
3296acc7838Sopenharmony_ci                let e_d: Option<&str> = $d.into();
3306acc7838Sopenharmony_ci                assert_eq!((v, d), ($v.to_string(), e_d.map(|s| s.into())));
3316acc7838Sopenharmony_ci            } else {
3326acc7838Sopenharmony_ci                panic!("{:?} didn't parse for version testing.", $s);
3336acc7838Sopenharmony_ci            }
3346acc7838Sopenharmony_ci        });
3356acc7838Sopenharmony_ci        ($f:expr, $s:expr => $v:expr, $d:expr) => ({
3366acc7838Sopenharmony_ci            let warn = "warning: invalid logging spec 'warning', ignoring it";
3376acc7838Sopenharmony_ci            let warn2 = "warning: sorry, something went wrong :(sad)";
3386acc7838Sopenharmony_ci            check_parse!(@ $f, $s => $v, $d);
3396acc7838Sopenharmony_ci            check_parse!(@ $f, &format!("{}\n{}", warn, $s) => $v, $d);
3406acc7838Sopenharmony_ci            check_parse!(@ $f, &format!("{}\n{}", warn2, $s) => $v, $d);
3416acc7838Sopenharmony_ci            check_parse!(@ $f, &format!("{}\n{}\n{}", warn, warn2, $s) => $v, $d);
3426acc7838Sopenharmony_ci            check_parse!(@ $f, &format!("{}\n{}\n{}", warn2, warn, $s) => $v, $d);
3436acc7838Sopenharmony_ci        })
3446acc7838Sopenharmony_ci    }
3456acc7838Sopenharmony_ci
3466acc7838Sopenharmony_ci    macro_rules! check_terse_parse {
3476acc7838Sopenharmony_ci        ($($s:expr => $v:expr, $d:expr,)+) => {$(
3486acc7838Sopenharmony_ci            check_parse!(version_and_date_from_rustc_version, $s => $v, $d);
3496acc7838Sopenharmony_ci        )+}
3506acc7838Sopenharmony_ci    }
3516acc7838Sopenharmony_ci
3526acc7838Sopenharmony_ci    macro_rules! check_verbose_parse {
3536acc7838Sopenharmony_ci        ($($s:expr => $v:expr, $d:expr,)+) => {$(
3546acc7838Sopenharmony_ci            check_parse!(version_and_date_from_rustc_verbose_version, $s => $v, $d);
3556acc7838Sopenharmony_ci        )+}
3566acc7838Sopenharmony_ci    }
3576acc7838Sopenharmony_ci
3586acc7838Sopenharmony_ci    #[test]
3596acc7838Sopenharmony_ci    fn test_version_parse() {
3606acc7838Sopenharmony_ci        check_terse_parse! {
3616acc7838Sopenharmony_ci            "rustc 1.18.0" => "1.18.0", None,
3626acc7838Sopenharmony_ci            "rustc 1.8.0" => "1.8.0", None,
3636acc7838Sopenharmony_ci            "rustc 1.20.0-nightly" => "1.20.0-nightly", None,
3646acc7838Sopenharmony_ci            "rustc 1.20" => "1.20", None,
3656acc7838Sopenharmony_ci            "rustc 1.3" => "1.3", None,
3666acc7838Sopenharmony_ci            "rustc 1" => "1", None,
3676acc7838Sopenharmony_ci            "rustc 1.5.1-beta" => "1.5.1-beta", None,
3686acc7838Sopenharmony_ci            "rustc 1.20.0 (2017-07-09)" => "1.20.0", Some("2017-07-09"),
3696acc7838Sopenharmony_ci            "rustc 1.20.0-dev (2017-07-09)" => "1.20.0-dev", Some("2017-07-09"),
3706acc7838Sopenharmony_ci            "rustc 1.20.0-nightly (d84693b93 2017-07-09)" => "1.20.0-nightly", Some("2017-07-09"),
3716acc7838Sopenharmony_ci            "rustc 1.20.0 (d84693b93 2017-07-09)" => "1.20.0", Some("2017-07-09"),
3726acc7838Sopenharmony_ci            "rustc 1.30.0-nightly (3bc2ca7e4 2018-09-20)" => "1.30.0-nightly", Some("2018-09-20"),
3736acc7838Sopenharmony_ci        };
3746acc7838Sopenharmony_ci    }
3756acc7838Sopenharmony_ci
3766acc7838Sopenharmony_ci    #[test]
3776acc7838Sopenharmony_ci    fn test_verbose_version_parse() {
3786acc7838Sopenharmony_ci        check_verbose_parse! {
3796acc7838Sopenharmony_ci            "rustc 1.0.0 (a59de37e9 2015-05-13) (built 2015-05-14)\n\
3806acc7838Sopenharmony_ci                binary: rustc\n\
3816acc7838Sopenharmony_ci                commit-hash: a59de37e99060162a2674e3ff45409ac73595c0e\n\
3826acc7838Sopenharmony_ci                commit-date: 2015-05-13\n\
3836acc7838Sopenharmony_ci                build-date: 2015-05-14\n\
3846acc7838Sopenharmony_ci                host: x86_64-unknown-linux-gnu\n\
3856acc7838Sopenharmony_ci                release: 1.0.0" => "1.0.0", Some("2015-05-13"),
3866acc7838Sopenharmony_ci
3876acc7838Sopenharmony_ci            "rustc 1.0.0 (a59de37e9 2015-05-13) (built 2015-05-14)\n\
3886acc7838Sopenharmony_ci                commit-hash: a59de37e99060162a2674e3ff45409ac73595c0e\n\
3896acc7838Sopenharmony_ci                commit-date: 2015-05-13\n\
3906acc7838Sopenharmony_ci                build-date: 2015-05-14\n\
3916acc7838Sopenharmony_ci                host: x86_64-unknown-linux-gnu\n\
3926acc7838Sopenharmony_ci                release: 1.0.0" => "1.0.0", Some("2015-05-13"),
3936acc7838Sopenharmony_ci
3946acc7838Sopenharmony_ci            "rustc 1.50.0 (cb75ad5db 2021-02-10)\n\
3956acc7838Sopenharmony_ci                binary: rustc\n\
3966acc7838Sopenharmony_ci                commit-hash: cb75ad5db02783e8b0222fee363c5f63f7e2cf5b\n\
3976acc7838Sopenharmony_ci                commit-date: 2021-02-10\n\
3986acc7838Sopenharmony_ci                host: x86_64-unknown-linux-gnu\n\
3996acc7838Sopenharmony_ci                release: 1.50.0" => "1.50.0", Some("2021-02-10"),
4006acc7838Sopenharmony_ci
4016acc7838Sopenharmony_ci            "rustc 1.52.0-nightly (234781afe 2021-03-07)\n\
4026acc7838Sopenharmony_ci                binary: rustc\n\
4036acc7838Sopenharmony_ci                commit-hash: 234781afe33d3f339b002f85f948046d8476cfc9\n\
4046acc7838Sopenharmony_ci                commit-date: 2021-03-07\n\
4056acc7838Sopenharmony_ci                host: x86_64-unknown-linux-gnu\n\
4066acc7838Sopenharmony_ci                release: 1.52.0-nightly\n\
4076acc7838Sopenharmony_ci                LLVM version: 12.0.0" => "1.52.0-nightly", Some("2021-03-07"),
4086acc7838Sopenharmony_ci
4096acc7838Sopenharmony_ci            "rustc 1.41.1\n\
4106acc7838Sopenharmony_ci                binary: rustc\n\
4116acc7838Sopenharmony_ci                commit-hash: unknown\n\
4126acc7838Sopenharmony_ci                commit-date: unknown\n\
4136acc7838Sopenharmony_ci                host: x86_64-unknown-linux-gnu\n\
4146acc7838Sopenharmony_ci                release: 1.41.1\n\
4156acc7838Sopenharmony_ci                LLVM version: 7.0" => "1.41.1", None,
4166acc7838Sopenharmony_ci
4176acc7838Sopenharmony_ci            "rustc 1.49.0\n\
4186acc7838Sopenharmony_ci                binary: rustc\n\
4196acc7838Sopenharmony_ci                commit-hash: unknown\n\
4206acc7838Sopenharmony_ci                commit-date: unknown\n\
4216acc7838Sopenharmony_ci                host: x86_64-unknown-linux-gnu\n\
4226acc7838Sopenharmony_ci                release: 1.49.0" => "1.49.0", None,
4236acc7838Sopenharmony_ci
4246acc7838Sopenharmony_ci            "rustc 1.50.0 (Fedora 1.50.0-1.fc33)\n\
4256acc7838Sopenharmony_ci                binary: rustc\n\
4266acc7838Sopenharmony_ci                commit-hash: unknown\n\
4276acc7838Sopenharmony_ci                commit-date: unknown\n\
4286acc7838Sopenharmony_ci                host: x86_64-unknown-linux-gnu\n\
4296acc7838Sopenharmony_ci                release: 1.50.0" => "1.50.0", None,
4306acc7838Sopenharmony_ci        };
4316acc7838Sopenharmony_ci    }
4326acc7838Sopenharmony_ci
4336acc7838Sopenharmony_ci    fn read_static(verbose: bool, channel: &str, minor: usize) -> String {
4346acc7838Sopenharmony_ci        use std::fs::File;
4356acc7838Sopenharmony_ci        use std::path::Path;
4366acc7838Sopenharmony_ci        use std::io::{BufReader, Read};
4376acc7838Sopenharmony_ci
4386acc7838Sopenharmony_ci        let subdir = if verbose { "verbose" } else { "terse" };
4396acc7838Sopenharmony_ci        let path = Path::new(STATIC_PATH)
4406acc7838Sopenharmony_ci            .join(channel)
4416acc7838Sopenharmony_ci            .join(subdir)
4426acc7838Sopenharmony_ci            .join(format!("rustc-1.{}.0", minor));
4436acc7838Sopenharmony_ci
4446acc7838Sopenharmony_ci        let file = File::open(path).unwrap();
4456acc7838Sopenharmony_ci        let mut buf_reader = BufReader::new(file);
4466acc7838Sopenharmony_ci        let mut contents = String::new();
4476acc7838Sopenharmony_ci        buf_reader.read_to_string(&mut contents).unwrap();
4486acc7838Sopenharmony_ci        contents
4496acc7838Sopenharmony_ci    }
4506acc7838Sopenharmony_ci
4516acc7838Sopenharmony_ci    static STATIC_PATH: &'static str = concat!(env!("CARGO_MANIFEST_DIR"), "/static");
4526acc7838Sopenharmony_ci
4536acc7838Sopenharmony_ci    static DATES: [&'static str; 51] = [
4546acc7838Sopenharmony_ci        "2015-05-13", "2015-06-19", "2015-08-03", "2015-09-15", "2015-10-27",
4556acc7838Sopenharmony_ci        "2015-12-04", "2016-01-19", "2016-02-29", "2016-04-11", "2016-05-18",
4566acc7838Sopenharmony_ci        "2016-07-03", "2016-08-15", "2016-09-23", "2016-11-07", "2016-12-16",
4576acc7838Sopenharmony_ci        "2017-01-19", "2017-03-10", "2017-04-24", "2017-06-06", "2017-07-17",
4586acc7838Sopenharmony_ci        "2017-08-27", "2017-10-09", "2017-11-20", "2018-01-01", "2018-02-12",
4596acc7838Sopenharmony_ci        "2018-03-25", "2018-05-07", "2018-06-19", "2018-07-30", "2018-09-11",
4606acc7838Sopenharmony_ci        "2018-10-24", "2018-12-04", "2019-01-16", "2019-02-28", "2019-04-10",
4616acc7838Sopenharmony_ci        "2019-05-20", "2019-07-03", "2019-08-13", "2019-09-23", "2019-11-04",
4626acc7838Sopenharmony_ci        "2019-12-16", "2020-01-27", "2020-03-09", "2020-04-20", "2020-06-01",
4636acc7838Sopenharmony_ci        "2020-07-13", "2020-08-24", "2020-10-07", "2020-11-16", "2020-12-29",
4646acc7838Sopenharmony_ci        "2021-02-10",
4656acc7838Sopenharmony_ci    ];
4666acc7838Sopenharmony_ci
4676acc7838Sopenharmony_ci    #[test]
4686acc7838Sopenharmony_ci    fn test_stable_compatibility() {
4696acc7838Sopenharmony_ci        if env::var_os("FORCE_STATIC").is_none() && fs::metadata(STATIC_PATH).is_err() {
4706acc7838Sopenharmony_ci            // We exclude `/static` when we package `version_check`, so don't
4716acc7838Sopenharmony_ci            // run if static files aren't present unless we know they should be.
4726acc7838Sopenharmony_ci            return;
4736acc7838Sopenharmony_ci        }
4746acc7838Sopenharmony_ci
4756acc7838Sopenharmony_ci        // Ensure we can parse all output from all Linux stable releases.
4766acc7838Sopenharmony_ci        for v in 0..DATES.len() {
4776acc7838Sopenharmony_ci            let (version, date) = (&format!("1.{}.0", v), Some(DATES[v]));
4786acc7838Sopenharmony_ci            check_terse_parse!(read_static(false, "stable", v) => version, date,);
4796acc7838Sopenharmony_ci            check_verbose_parse!(read_static(true, "stable", v) => version, date,);
4806acc7838Sopenharmony_ci        }
4816acc7838Sopenharmony_ci    }
4826acc7838Sopenharmony_ci
4836acc7838Sopenharmony_ci    #[test]
4846acc7838Sopenharmony_ci    fn test_parse_current() {
4856acc7838Sopenharmony_ci        let (version, channel) = (::Version::read(), ::Channel::read());
4866acc7838Sopenharmony_ci        assert!(version.is_some());
4876acc7838Sopenharmony_ci        assert!(channel.is_some());
4886acc7838Sopenharmony_ci
4896acc7838Sopenharmony_ci        if let Ok(known_channel) = env::var("KNOWN_CHANNEL") {
4906acc7838Sopenharmony_ci            assert_eq!(channel, ::Channel::parse(&known_channel));
4916acc7838Sopenharmony_ci        }
4926acc7838Sopenharmony_ci    }
4936acc7838Sopenharmony_ci}
494