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