Fix env and add tests.

I updated env to use the nightly build. I also added several tests.
This commit is contained in:
Joseph Crail 2015-05-06 23:59:58 -04:00
parent 1f2b68251f
commit 500bbbfa83
2 changed files with 88 additions and 34 deletions

56
src/env/env.rs vendored
View file

@ -1,5 +1,4 @@
#![crate_name = "env"] #![crate_name = "env"]
#![feature(core, old_io, os)]
/* /*
* This file is part of the uutils coreutils package. * This file is part of the uutils coreutils package.
@ -13,7 +12,9 @@
/* last synced with: env (GNU coreutils) 8.13 */ /* last synced with: env (GNU coreutils) 8.13 */
#![allow(non_camel_case_types)] #![allow(non_camel_case_types)]
#![feature(box_syntax)]
use std::env;
use std::process::Command;
struct options { struct options {
ignore_env: bool, ignore_env: bool,
@ -42,28 +43,22 @@ fn version() {
// print name=value env pairs on screen // print name=value env pairs on screen
// if null is true, separate pairs with a \0, \n otherwise // if null is true, separate pairs with a \0, \n otherwise
fn print_env(null: bool) { fn print_env(null: bool) {
let env = std::os::env(); for (n, v) in env::vars() {
print!("{}={}{}", n, v, if null { '\0' } else { '\n' });
for &(ref n, ref v) in env.iter() {
print!("{}={}{}",
n.as_slice(),
v.as_slice(),
if null { '\0' } else { '\n' }
);
} }
} }
pub fn uumain(args: Vec<String>) -> i32 { pub fn uumain(args: Vec<String>) -> i32 {
let prog = args[0].as_slice(); let prog = &args[0];
// to handle arguments the same way than GNU env, we can't use getopts // to handle arguments the same way than GNU env, we can't use getopts
let mut opts = box options { let mut opts = Box::new(options {
ignore_env: false, ignore_env: false,
null: false, null: false,
unsets: vec!(), unsets: vec!(),
sets: vec!(), sets: vec!(),
program: vec!() program: vec!()
}; });
let mut wait_cmd = false; let mut wait_cmd = false;
let mut iter = args.iter(); let mut iter = args.iter();
@ -78,7 +73,7 @@ pub fn uumain(args: Vec<String>) -> i32 {
if wait_cmd { if wait_cmd {
// we still accept NAME=VAL here but not other options // we still accept NAME=VAL here but not other options
let mut sp = opt.as_slice().splitn(1, '='); let mut sp = opt.splitn(2, '=');
let name = sp.next(); let name = sp.next();
let value = sp.next(); let value = sp.next();
@ -92,8 +87,8 @@ pub fn uumain(args: Vec<String>) -> i32 {
break; break;
} }
} }
} else if opt.as_slice().starts_with("--") { } else if opt.starts_with("--") {
match opt.as_slice() { match opt.as_ref() {
"--help" => { usage(prog); return 0; } "--help" => { usage(prog); return 0; }
"--version" => { version(); return 0; } "--version" => { version(); return 0; }
@ -103,7 +98,7 @@ pub fn uumain(args: Vec<String>) -> i32 {
let var = iter.next(); let var = iter.next();
match var { match var {
None => println!("{}: this option requires an argument: {}", prog, opt.as_slice()), None => println!("{}: this option requires an argument: {}", prog, opt),
Some(s) => opts.unsets.push(s.to_string()) Some(s) => opts.unsets.push(s.to_string())
} }
} }
@ -114,7 +109,7 @@ pub fn uumain(args: Vec<String>) -> i32 {
return 1; return 1;
} }
} }
} else if opt.as_slice().starts_with("-") { } else if opt.starts_with("-") {
if opt.len() == 0 { if opt.len() == 0 {
// implies -i and stop parsing opts // implies -i and stop parsing opts
wait_cmd = true; wait_cmd = true;
@ -122,7 +117,7 @@ pub fn uumain(args: Vec<String>) -> i32 {
continue; continue;
} }
let mut chars = opt.as_slice().chars(); let mut chars = opt.chars();
chars.next(); chars.next();
for c in chars { for c in chars {
@ -136,7 +131,7 @@ pub fn uumain(args: Vec<String>) -> i32 {
let var = iter.next(); let var = iter.next();
match var { match var {
None => println!("{}: this option requires an argument: {}", prog, opt.as_slice()), None => println!("{}: this option requires an argument: {}", prog, opt),
Some(s) => opts.unsets.push(s.to_string()) Some(s) => opts.unsets.push(s.to_string())
} }
} }
@ -149,7 +144,7 @@ pub fn uumain(args: Vec<String>) -> i32 {
} }
} else { } else {
// is it a NAME=VALUE like opt ? // is it a NAME=VALUE like opt ?
let mut sp = opt.as_slice().splitn(1, '='); let mut sp = opt.splitn(2, "=");
let name = sp.next(); let name = sp.next();
let value = sp.next(); let value = sp.next();
@ -175,32 +170,25 @@ pub fn uumain(args: Vec<String>) -> i32 {
opts.program.push(opt.to_string()); opts.program.push(opt.to_string());
} }
let env = std::os::env();
if opts.ignore_env { if opts.ignore_env {
for &(ref name, _) in env.iter() { for (ref name, _) in env::vars() {
std::os::unsetenv(name.as_slice()) env::remove_var(name);
} }
} }
for name in opts.unsets.iter() { for name in opts.unsets.iter() {
std::os::unsetenv((name).as_slice()) env::remove_var(name);
} }
for &(ref name, ref val) in opts.sets.iter() { for &(ref name, ref val) in opts.sets.iter() {
std::os::setenv(name.as_slice(), val.as_slice()) env::set_var(name, val);
} }
if opts.program.len() >= 1 { if opts.program.len() >= 1 {
use std::old_io::process::{Command, InheritFd};
let prog = opts.program[0].clone(); let prog = opts.program[0].clone();
let args = &opts.program[1..]; let args = &opts.program[1..];
match Command::new(prog).args(args).stdin(InheritFd(0)).stdout(InheritFd(1)).stderr(InheritFd(2)).status() { match Command::new(prog).args(args).status() {
Ok(exit) => Ok(exit) => return if exit.success() { 0 } else { exit.code().unwrap() },
return match exit {
std::old_io::process::ExitStatus(s) => s as i32,
_ => 1
},
Err(_) => return 1 Err(_) => return 1
} }
} else { } else {

66
test/env.rs Normal file
View file

@ -0,0 +1,66 @@
use std::process::Command;
use std::str;
static PROGNAME: &'static str = "./env";
#[test]
fn test_single_name_value_pair() {
let po = Command::new(PROGNAME)
.arg("FOO=bar")
.output()
.unwrap_or_else(|err| panic!("{}", err));
let out = str::from_utf8(&po.stdout[..]).unwrap();
assert!(out.lines_any().any(|line| line == "FOO=bar"));
}
#[test]
fn test_multiple_name_value_pairs() {
let po = Command::new(PROGNAME)
.arg("FOO=bar")
.arg("ABC=xyz")
.output()
.unwrap_or_else(|err| panic!("{}", err));
let out = str::from_utf8(&po.stdout[..]).unwrap();
assert_eq!(out.lines_any().filter(|&line| line == "FOO=bar" || line == "ABC=xyz").count(), 2);
}
#[test]
fn test_ignore_environment() {
let po = Command::new(PROGNAME)
.arg("-i")
.output()
.unwrap_or_else(|err| panic!("{}", err));
let out = str::from_utf8(&po.stdout[..]).unwrap();
assert_eq!(out, "");
}
#[test]
fn test_null_delimiter() {
let po = Command::new(PROGNAME)
.arg("-i")
.arg("--null")
.arg("FOO=bar")
.arg("ABC=xyz")
.output()
.unwrap_or_else(|err| panic!("{}", err));
let out = str::from_utf8(&po.stdout[..]).unwrap();
assert_eq!(out, "FOO=bar\0ABC=xyz\0");
}
#[test]
fn test_unset_variable() {
// This test depends on the HOME variable being pre-defined by the
// default shell
let po = Command::new(PROGNAME)
.arg("-u")
.arg("HOME")
.output()
.unwrap_or_else(|err| panic!("{}", err));
let out = str::from_utf8(&po.stdout[..]).unwrap();
assert_eq!(out.lines_any().any(|line| line.starts_with("HOME=")), false);
}