283 lines
7.6 KiB
Rust
283 lines
7.6 KiB
Rust
use std::env::*;
|
|
use std::ffi::{OsStr, OsString};
|
|
use std::path::Path;
|
|
|
|
use rand::distributions::{Alphanumeric, DistString};
|
|
|
|
mod common;
|
|
use std::thread;
|
|
|
|
use common::test_rng;
|
|
|
|
#[track_caller]
|
|
fn make_rand_name() -> OsString {
|
|
let n = format!("TEST{}", Alphanumeric.sample_string(&mut test_rng(), 10));
|
|
let n = OsString::from(n);
|
|
assert!(var_os(&n).is_none());
|
|
n
|
|
}
|
|
|
|
fn eq(a: Option<OsString>, b: Option<&str>) {
|
|
assert_eq!(a.as_ref().map(|s| &**s), b.map(OsStr::new).map(|s| &*s));
|
|
}
|
|
|
|
#[test]
|
|
fn test_set_var() {
|
|
let n = make_rand_name();
|
|
set_var(&n, "VALUE");
|
|
eq(var_os(&n), Some("VALUE"));
|
|
}
|
|
|
|
#[test]
|
|
fn test_remove_var() {
|
|
let n = make_rand_name();
|
|
set_var(&n, "VALUE");
|
|
remove_var(&n);
|
|
eq(var_os(&n), None);
|
|
}
|
|
|
|
#[test]
|
|
fn test_set_var_overwrite() {
|
|
let n = make_rand_name();
|
|
set_var(&n, "1");
|
|
set_var(&n, "2");
|
|
eq(var_os(&n), Some("2"));
|
|
set_var(&n, "");
|
|
eq(var_os(&n), Some(""));
|
|
}
|
|
|
|
#[test]
|
|
#[cfg_attr(target_os = "emscripten", ignore)]
|
|
fn test_var_big() {
|
|
let mut s = "".to_string();
|
|
let mut i = 0;
|
|
while i < 100 {
|
|
s.push_str("aaaaaaaaaa");
|
|
i += 1;
|
|
}
|
|
let n = make_rand_name();
|
|
set_var(&n, &s);
|
|
eq(var_os(&n), Some(&s));
|
|
}
|
|
|
|
#[test]
|
|
#[cfg_attr(target_os = "emscripten", ignore)]
|
|
fn test_env_set_get_huge() {
|
|
let n = make_rand_name();
|
|
let s = "x".repeat(10000);
|
|
set_var(&n, &s);
|
|
eq(var_os(&n), Some(&s));
|
|
remove_var(&n);
|
|
eq(var_os(&n), None);
|
|
}
|
|
|
|
#[test]
|
|
fn test_env_set_var() {
|
|
let n = make_rand_name();
|
|
|
|
let mut e = vars_os();
|
|
set_var(&n, "VALUE");
|
|
assert!(!e.any(|(k, v)| { &*k == &*n && &*v == "VALUE" }));
|
|
|
|
assert!(vars_os().any(|(k, v)| { &*k == &*n && &*v == "VALUE" }));
|
|
}
|
|
|
|
#[test]
|
|
#[cfg_attr(not(any(unix, windows)), ignore, allow(unused))]
|
|
#[allow(deprecated)]
|
|
fn env_home_dir() {
|
|
use std::path::PathBuf;
|
|
|
|
fn var_to_os_string(var: Result<String, VarError>) -> Option<OsString> {
|
|
match var {
|
|
Ok(var) => Some(OsString::from(var)),
|
|
Err(VarError::NotUnicode(var)) => Some(var),
|
|
_ => None,
|
|
}
|
|
}
|
|
|
|
cfg_if::cfg_if! {
|
|
if #[cfg(unix)] {
|
|
let oldhome = var_to_os_string(var("HOME"));
|
|
|
|
set_var("HOME", "/home/MountainView");
|
|
assert_eq!(home_dir(), Some(PathBuf::from("/home/MountainView")));
|
|
|
|
remove_var("HOME");
|
|
if cfg!(target_os = "android") {
|
|
assert!(home_dir().is_none());
|
|
} else {
|
|
// When HOME is not set, some platforms return `None`,
|
|
// but others return `Some` with a default.
|
|
// Just check that it is not "/home/MountainView".
|
|
assert_ne!(home_dir(), Some(PathBuf::from("/home/MountainView")));
|
|
}
|
|
|
|
if let Some(oldhome) = oldhome { set_var("HOME", oldhome); }
|
|
} else if #[cfg(windows)] {
|
|
let oldhome = var_to_os_string(var("HOME"));
|
|
let olduserprofile = var_to_os_string(var("USERPROFILE"));
|
|
|
|
remove_var("HOME");
|
|
remove_var("USERPROFILE");
|
|
|
|
assert!(home_dir().is_some());
|
|
|
|
set_var("HOME", "/home/PaloAlto");
|
|
assert_ne!(home_dir(), Some(PathBuf::from("/home/PaloAlto")), "HOME must not be used");
|
|
|
|
set_var("USERPROFILE", "/home/MountainView");
|
|
assert_eq!(home_dir(), Some(PathBuf::from("/home/MountainView")));
|
|
|
|
remove_var("HOME");
|
|
|
|
assert_eq!(home_dir(), Some(PathBuf::from("/home/MountainView")));
|
|
|
|
set_var("USERPROFILE", "");
|
|
assert_ne!(home_dir(), Some(PathBuf::from("")), "Empty USERPROFILE must be ignored");
|
|
|
|
remove_var("USERPROFILE");
|
|
|
|
if let Some(oldhome) = oldhome { set_var("HOME", oldhome); }
|
|
if let Some(olduserprofile) = olduserprofile { set_var("USERPROFILE", olduserprofile); }
|
|
}
|
|
}
|
|
}
|
|
|
|
#[test] // miri shouldn't detect any data race in this fn
|
|
#[cfg_attr(any(not(miri), target_os = "emscripten"), ignore)]
|
|
fn test_env_get_set_multithreaded() {
|
|
let getter = thread::spawn(|| {
|
|
for _ in 0..100 {
|
|
let _ = var_os("foo");
|
|
}
|
|
});
|
|
|
|
let setter = thread::spawn(|| {
|
|
for _ in 0..100 {
|
|
set_var("foo", "bar");
|
|
}
|
|
});
|
|
|
|
let _ = getter.join();
|
|
let _ = setter.join();
|
|
}
|
|
|
|
#[test]
|
|
#[cfg_attr(any(target_os = "emscripten", target_os = "wasi", target_env = "sgx"), ignore)]
|
|
fn test_self_exe_path() {
|
|
let path = current_exe();
|
|
assert!(path.is_ok());
|
|
let path = path.unwrap();
|
|
|
|
// Hard to test this function
|
|
assert!(path.is_absolute());
|
|
}
|
|
|
|
#[test]
|
|
fn test() {
|
|
assert!((!Path::new("test-path").is_absolute()));
|
|
|
|
#[cfg(not(target_env = "sgx"))]
|
|
current_dir().unwrap();
|
|
}
|
|
|
|
#[test]
|
|
#[cfg(windows)]
|
|
fn split_paths_windows() {
|
|
use std::path::PathBuf;
|
|
|
|
fn check_parse(unparsed: &str, parsed: &[&str]) -> bool {
|
|
split_paths(unparsed).collect::<Vec<_>>()
|
|
== parsed.iter().map(|s| PathBuf::from(*s)).collect::<Vec<_>>()
|
|
}
|
|
|
|
assert!(check_parse("", &mut [""]));
|
|
assert!(check_parse(r#""""#, &mut [""]));
|
|
assert!(check_parse(";;", &mut ["", "", ""]));
|
|
assert!(check_parse(r"c:\", &mut [r"c:\"]));
|
|
assert!(check_parse(r"c:\;", &mut [r"c:\", ""]));
|
|
assert!(check_parse(r"c:\;c:\Program Files\", &mut [r"c:\", r"c:\Program Files\"]));
|
|
assert!(check_parse(r#"c:\;c:\"foo"\"#, &mut [r"c:\", r"c:\foo\"]));
|
|
assert!(check_parse(r#"c:\;c:\"foo;bar"\;c:\baz"#, &mut [r"c:\", r"c:\foo;bar\", r"c:\baz"]));
|
|
}
|
|
|
|
#[test]
|
|
#[cfg(unix)]
|
|
fn split_paths_unix() {
|
|
use std::path::PathBuf;
|
|
|
|
fn check_parse(unparsed: &str, parsed: &[&str]) -> bool {
|
|
split_paths(unparsed).collect::<Vec<_>>()
|
|
== parsed.iter().map(|s| PathBuf::from(*s)).collect::<Vec<_>>()
|
|
}
|
|
|
|
assert!(check_parse("", &mut [""]));
|
|
assert!(check_parse("::", &mut ["", "", ""]));
|
|
assert!(check_parse("/", &mut ["/"]));
|
|
assert!(check_parse("/:", &mut ["/", ""]));
|
|
assert!(check_parse("/:/usr/local", &mut ["/", "/usr/local"]));
|
|
}
|
|
|
|
#[test]
|
|
#[cfg(unix)]
|
|
fn join_paths_unix() {
|
|
use std::ffi::OsStr;
|
|
|
|
fn test_eq(input: &[&str], output: &str) -> bool {
|
|
&*join_paths(input.iter().cloned()).unwrap() == OsStr::new(output)
|
|
}
|
|
|
|
assert!(test_eq(&[], ""));
|
|
assert!(test_eq(&["/bin", "/usr/bin", "/usr/local/bin"], "/bin:/usr/bin:/usr/local/bin"));
|
|
assert!(test_eq(&["", "/bin", "", "", "/usr/bin", ""], ":/bin:::/usr/bin:"));
|
|
assert!(join_paths(["/te:st"].iter().cloned()).is_err());
|
|
}
|
|
|
|
#[test]
|
|
#[cfg(windows)]
|
|
fn join_paths_windows() {
|
|
use std::ffi::OsStr;
|
|
|
|
fn test_eq(input: &[&str], output: &str) -> bool {
|
|
&*join_paths(input.iter().cloned()).unwrap() == OsStr::new(output)
|
|
}
|
|
|
|
assert!(test_eq(&[], ""));
|
|
assert!(test_eq(&[r"c:\windows", r"c:\"], r"c:\windows;c:\"));
|
|
assert!(test_eq(&["", r"c:\windows", "", "", r"c:\", ""], r";c:\windows;;;c:\;"));
|
|
assert!(test_eq(&[r"c:\te;st", r"c:\"], r#""c:\te;st";c:\"#));
|
|
assert!(join_paths([r#"c:\te"st"#].iter().cloned()).is_err());
|
|
}
|
|
|
|
#[test]
|
|
fn args_debug() {
|
|
assert_eq!(
|
|
format!("Args {{ inner: {:?} }}", args().collect::<Vec<_>>()),
|
|
format!("{:?}", args())
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
fn args_os_debug() {
|
|
assert_eq!(
|
|
format!("ArgsOs {{ inner: {:?} }}", args_os().collect::<Vec<_>>()),
|
|
format!("{:?}", args_os())
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
fn vars_debug() {
|
|
assert_eq!(
|
|
format!("Vars {{ inner: {:?} }}", vars().collect::<Vec<_>>()),
|
|
format!("{:?}", vars())
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
fn vars_os_debug() {
|
|
assert_eq!(
|
|
format!("VarsOs {{ inner: {:?} }}", vars_os().collect::<Vec<_>>()),
|
|
format!("{:?}", vars_os())
|
|
);
|
|
}
|