2015-04-28 01:10:37 -04:00
|
|
|
/* Pirate - A command-line arrrrguments parser, written in Rust.
|
|
|
|
* Copyright (C) 2015 Zachary Dziura
|
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
pub mod errors;
|
|
|
|
pub mod matches;
|
|
|
|
mod opts;
|
|
|
|
|
|
|
|
use std::env::Args;
|
|
|
|
|
2015-04-28 18:40:22 +00:00
|
|
|
pub use errors::{Error, ErrorKind};
|
2015-04-28 01:10:37 -04:00
|
|
|
pub use matches::Matches;
|
|
|
|
use opts::Opts;
|
|
|
|
|
2015-04-30 15:14:34 +00:00
|
|
|
pub fn parse(mut args: Args, options: &[&'static str]) -> Result<Matches, Error> {
|
2015-04-28 01:10:37 -04:00
|
|
|
let mut matches: Matches = Matches::new();
|
|
|
|
|
|
|
|
let mut opts: Opts = opts(options); // Jesus, this is redundant...
|
|
|
|
|
|
|
|
args.next(); // Remove the program name from the list of program arguments
|
|
|
|
|
|
|
|
let mut next_arg = args.next();
|
|
|
|
while next_arg.is_some() {
|
|
|
|
let mut current_arg = next_arg.unwrap();
|
|
|
|
let arg: String;
|
|
|
|
|
|
|
|
if ¤t_arg[..1] == "-" { // Probably a opt
|
|
|
|
if current_arg.len() == 2 { // Short form opt
|
|
|
|
arg = String::from(¤t_arg[1..]);
|
|
|
|
} else { // Assuming it's a long form opt
|
|
|
|
//TODO: Handle cases where it may be a opt group
|
|
|
|
arg = String::from(¤t_arg[2..]);
|
|
|
|
}
|
|
|
|
|
|
|
|
if opts.contains_opt(&arg) {
|
|
|
|
let has_arg: bool = *opts.get_opt(&arg).unwrap();
|
|
|
|
|
|
|
|
if has_arg {
|
|
|
|
// NOTE: The corresponding arg MUST be immediately following
|
|
|
|
current_arg = match args.next() {
|
|
|
|
None => {
|
2015-04-28 18:40:22 +00:00
|
|
|
return Err(Error::new(ErrorKind::MissingArgument, arg));
|
2015-04-28 01:10:37 -04:00
|
|
|
},
|
|
|
|
Some(a) => a
|
|
|
|
};
|
|
|
|
|
|
|
|
matches.insert(&arg, ¤t_arg);
|
|
|
|
} else {
|
|
|
|
matches.insert(&arg, "");
|
|
|
|
}
|
|
|
|
} else {
|
2015-05-27 14:21:03 -04:00
|
|
|
return Err(Error::new(ErrorKind::InvalidArgument, arg));
|
2015-04-28 01:10:37 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
} else { // Probably a required arg
|
|
|
|
let arg_name: String = opts.get_arg().unwrap();
|
|
|
|
matches.insert(&arg_name, ¤t_arg);
|
|
|
|
}
|
|
|
|
|
|
|
|
next_arg = args.next();
|
|
|
|
}
|
|
|
|
|
2015-04-30 15:14:34 +00:00
|
|
|
match opts.arg_len() {
|
|
|
|
0 => Ok(matches),
|
|
|
|
_ => Err(Error::new(ErrorKind::MissingArgument, opts.get_arg().unwrap())),
|
|
|
|
}
|
2015-04-28 01:10:37 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
fn opts(opts: &[&'static str]) -> Opts {
|
|
|
|
let mut options = Opts::new();
|
|
|
|
|
|
|
|
for opt in opts.iter() {
|
|
|
|
let is_arg: bool = match &opt[..1] {
|
|
|
|
":" => true,
|
|
|
|
_ => false
|
|
|
|
};
|
|
|
|
|
|
|
|
let has_arg: bool = match &opt[(opt.len() - 1)..] {
|
|
|
|
":" => true,
|
|
|
|
_ => false
|
|
|
|
};
|
|
|
|
|
|
|
|
if is_arg {
|
|
|
|
options.insert_arg(&opt[1..]);
|
|
|
|
} else {
|
|
|
|
let option: &str;
|
|
|
|
|
|
|
|
if has_arg {
|
|
|
|
option = &opt[..(opt.len() - 1)];
|
|
|
|
} else {
|
|
|
|
option = *opt;
|
|
|
|
}
|
|
|
|
|
|
|
|
for form in option.split("/") {
|
|
|
|
options.insert_opt(form, has_arg);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Push the obligatory "-h/--help" options
|
|
|
|
options.insert_opt("h", false);
|
|
|
|
options.insert_opt("help", false);
|
|
|
|
|
|
|
|
options
|
|
|
|
}
|