Allow for different types of failure

This commit is contained in:
Benjamin Sago 2015-01-12 22:14:27 +01:00
parent 3f08836c3f
commit 23e5d3ce8d
2 changed files with 45 additions and 27 deletions

View File

@ -7,11 +7,12 @@ extern crate users;
use std::io::FileType; use std::io::FileType;
use std::io::fs; use std::io::fs;
use std::os; use std::os::{args, set_exit_status};
use dir::Dir; use dir::Dir;
use file::File; use file::File;
use options::Options; use options::Options;
use options::Error::*;
pub mod column; pub mod column;
pub mod dir; pub mod dir;
@ -21,15 +22,6 @@ pub mod options;
pub mod output; pub mod output;
pub mod term; pub mod term;
fn main() {
let args: Vec<String> = os::args();
match Options::getopts(args) {
Err(error_code) => os::set_exit_status(error_code),
Ok(options) => exa(&options),
};
}
fn exa(options: &Options) { fn exa(options: &Options) {
let mut dirs: Vec<String> = vec![]; let mut dirs: Vec<String> = vec![];
let mut files: Vec<File> = vec![]; let mut files: Vec<File> = vec![];
@ -92,3 +84,19 @@ fn exa(options: &Options) {
}; };
} }
} }
fn main() {
let args: Vec<String> = args();
match Options::getopts(args) {
Ok(options) => exa(&options),
Err(Help(text)) => {
println!("{}", text);
set_exit_status(2);
},
Err(InvalidOptions(e)) => {
println!("{}", e);
set_exit_status(3);
},
};
}

View File

@ -17,18 +17,22 @@ pub enum SortField {
impl Copy for SortField { } impl Copy for SortField { }
impl SortField { impl SortField {
fn from_word(word: String) -> SortField { fn from_word(word: String) -> Result<SortField, Error> {
match word.as_slice() { match word.as_slice() {
"name" => SortField::Name, "name" => Ok(SortField::Name),
"size" => SortField::Size, "size" => Ok(SortField::Size),
"ext" => SortField::Extension, "ext" => Ok(SortField::Extension),
"none" => SortField::Unsorted, "none" => Ok(SortField::Unsorted),
"inode" => SortField::FileInode, "inode" => Ok(SortField::FileInode),
_ => panic!("Invalid sorting order"), field => Err(no_sort_field(field))
} }
} }
} }
fn no_sort_field(field: &str) -> Error {
Error::InvalidOptions(getopts::Fail::UnrecognizedOption(format!("--sort {}", field)))
}
pub struct Options { pub struct Options {
pub list_dirs: bool, pub list_dirs: bool,
path_strs: Vec<String>, path_strs: Vec<String>,
@ -38,9 +42,14 @@ pub struct Options {
pub view: View, pub view: View,
} }
pub enum Error {
InvalidOptions(getopts::Fail),
Help(String),
}
impl Options { impl Options {
pub fn getopts(args: Vec<String>) -> Result<Options, isize> { pub fn getopts(args: Vec<String>) -> Result<Options, Error> {
let opts = [ let opts = &[
getopts::optflag("1", "oneline", "display one entry per line"), getopts::optflag("1", "oneline", "display one entry per line"),
getopts::optflag("a", "all", "show dot-files"), getopts::optflag("a", "all", "show dot-files"),
getopts::optflag("b", "binary", "use binary prefixes in file sizes"), getopts::optflag("b", "binary", "use binary prefixes in file sizes"),
@ -58,25 +67,26 @@ impl Options {
getopts::optflag("?", "help", "show list of command-line options"), getopts::optflag("?", "help", "show list of command-line options"),
]; ];
let matches = match getopts::getopts(args.tail(), &opts) { let matches = match getopts::getopts(args.tail(), opts) {
Ok(m) => m, Ok(m) => m,
Err(e) => { Err(e) => return Err(Error::InvalidOptions(e)),
println!("Invalid options: {}", e);
return Err(1);
}
}; };
if matches.opt_present("help") { if matches.opt_present("help") {
println!("exa - ls with more features\n\n{}", getopts::usage("Usage:\n exa [options] [files...]", &opts)); return Err(Error::Help(getopts::usage("Usage:\n exa [options] [files...]", opts)));
return Err(2);
} }
let sort_field = match matches.opt_str("sort") {
Some(word) => try!(SortField::from_word(word)),
None => SortField::Name,
};
Ok(Options { Ok(Options {
list_dirs: matches.opt_present("list-dirs"), list_dirs: matches.opt_present("list-dirs"),
path_strs: if matches.free.is_empty() { vec![ ".".to_string() ] } else { matches.free.clone() }, path_strs: if matches.free.is_empty() { vec![ ".".to_string() ] } else { matches.free.clone() },
reverse: matches.opt_present("reverse"), reverse: matches.opt_present("reverse"),
show_invisibles: matches.opt_present("all"), show_invisibles: matches.opt_present("all"),
sort_field: matches.opt_str("sort").map(|word| SortField::from_word(word)).unwrap_or(SortField::Name), sort_field: sort_field,
view: Options::view(&matches), view: Options::view(&matches),
}) })
} }