clap/src/errors.rs

832 lines
30 KiB
Rust
Raw Normal View History

use std::process;
use std::error::Error as StdError;
use std::fmt as std_fmt;
use std::fmt::Display;
use std::io::{self, Write};
use std::convert::From;
use std::result::Result as StdResult;
use fmt;
use suggestions;
use args::any_arg::AnyArg;
/// Short hand for [`Result`] type
/// [`Result`]: https://doc.rust-lang.org/std/result/enum.Result.html
pub type Result<T> = StdResult<T, Error>;
2016-01-26 19:02:10 +00:00
/// Command line argument parser kind of error
2015-10-31 05:19:24 +00:00
#[derive(Debug, Copy, Clone, PartialEq)]
pub enum ErrorKind {
/// Occurs when an [`Arg`] has a set of possible values, and the user provides a value which
/// isn't in that set.
///
/// # Examples
///
2016-01-26 19:02:10 +00:00
/// ```rust
/// # use clap::{App, Arg, ErrorKind};
/// let result = App::new("myprog")
/// .arg(Arg::with_name("speed")
/// .possible_value("fast")
/// .possible_value("slow"))
/// .get_matches_from_safe(vec!["myprog", "other"]);
/// assert!(result.is_err());
/// assert_eq!(result.unwrap_err().kind, ErrorKind::InvalidValue);
2015-09-05 22:12:46 +00:00
/// ```
/// [`Arg`]: ./struct.Arg.html
InvalidValue,
2016-01-29 02:51:50 +00:00
/// Occurs when a user provides a flag, option, or argument which wasn't defined.
///
/// # Examples
///
2016-01-26 19:02:10 +00:00
/// ```rust
/// # use clap::{App, Arg, ErrorKind};
/// let result = App::new("myprog")
/// .arg(Arg::from_usage("--flag 'some flag'"))
/// .get_matches_from_safe(vec!["myprog", "--other"]);
/// assert!(result.is_err());
/// assert_eq!(result.unwrap_err().kind, ErrorKind::UnknownArgument);
2015-09-05 22:12:46 +00:00
/// ```
UnknownArgument,
/// Occurs when the user provids an unrecognized [`SubCommand`] which meets the threshold for
/// being similar enough to an existing subcommand so as to not cause the more general
/// [`UnknownArgument`] error.
///
/// # Examples
///
2016-01-26 19:02:10 +00:00
/// ```rust
/// # use clap::{App, Arg, ErrorKind, SubCommand};
/// let result = App::new("myprog")
2015-11-01 14:02:37 +00:00
/// .subcommand(SubCommand::with_name("config")
/// .about("Used for configuration")
/// .arg(Arg::with_name("config_file")
/// .help("The configuration file to use")
/// .index(1)))
2016-01-26 19:02:10 +00:00
/// .get_matches_from_safe(vec!["myprog", "confi"]);
/// assert!(result.is_err());
/// assert_eq!(result.unwrap_err().kind, ErrorKind::InvalidSubcommand);
2015-09-05 22:12:46 +00:00
/// ```
/// [`SubCommand`]: ./struct.SubCommand.html
/// [`UnknownArgument`]: ./enum.ErrorKind.html#variant.UnknownArgument
InvalidSubcommand,
/// Occurs when the user provids an unrecognized [`SubCommand`] which does not meet the
/// threshold for being similar enough to an existing subcommand so as to not cause the more
/// detailed [`InvalidSubcommand`] error.
///
/// This error typically happens when passing additional subcommand names to the `help`
/// subcommand. Otherwise, the more general [`UnknownArgument`] error is used.
///
/// # Examples
///
/// ```rust
/// # use clap::{App, Arg, ErrorKind, SubCommand};
/// let result = App::new("myprog")
/// .subcommand(SubCommand::with_name("config")
/// .about("Used for configuration")
/// .arg(Arg::with_name("config_file")
/// .help("The configuration file to use")
/// .index(1)))
/// .get_matches_from_safe(vec!["myprog", "help", "nothing"]);
/// assert!(result.is_err());
/// assert_eq!(result.unwrap_err().kind, ErrorKind::UnrecognizedSubcommand);
/// ```
/// [`SubCommand`]: ./struct.SubCommand.html
/// [`InvalidSubcommand`]: ./enum.ErrorKind.html#variant.InvalidSubcommand
/// [`UnknownArgument`]: ./enum.ErrorKind.html#variant.UnknownArgument
UnrecognizedSubcommand,
2016-01-29 02:51:50 +00:00
/// Occurs when the user provides an empty value for an option that does not allow empty
/// values.
///
/// # Examples
///
2016-01-26 19:02:10 +00:00
/// ```rust
/// # use clap::{App, Arg, ErrorKind};
2016-01-26 19:02:10 +00:00
/// let res = App::new("myprog")
/// .arg(Arg::with_name("color")
2016-01-26 19:02:10 +00:00
/// .long("color")
/// .empty_values(false))
2016-01-26 19:02:10 +00:00
/// .get_matches_from_safe(vec!["myprog", "--color="]);
/// assert!(res.is_err());
/// assert_eq!(res.unwrap_err().kind, ErrorKind::EmptyValue);
2015-09-05 22:12:46 +00:00
/// ```
EmptyValue,
/// Occurs when the user provides a value for an argument with a custom validation and the
/// value fails that validation.
///
/// # Examples
///
2016-01-26 19:02:10 +00:00
/// ```rust
/// # use clap::{App, Arg, ErrorKind};
/// fn is_numeric(val: String) -> Result<(), String> {
/// match val.parse::<i64>() {
/// Ok(..) => Ok(()),
/// Err(..) => Err(String::from("Value wasn't a number!")),
/// }
/// }
///
/// let result = App::new("myprog")
/// .arg(Arg::with_name("num")
/// .validator(is_numeric))
/// .get_matches_from_safe(vec!["myprog", "NotANumber"]);
/// assert!(result.is_err());
/// assert_eq!(result.unwrap_err().kind, ErrorKind::ValueValidation);
/// ```
ValueValidation,
/// Occurs when a user provides more values for an argument than were defined by setting
/// [`Arg::max_values`].
///
/// # Examples
///
2016-01-26 19:02:10 +00:00
/// ```rust
/// # use clap::{App, Arg, ErrorKind};
/// let result = App::new("myprog")
2016-01-26 19:02:10 +00:00
/// .arg(Arg::with_name("arg")
/// .multiple(true)
/// .max_values(2))
2016-01-26 19:02:10 +00:00
/// .get_matches_from_safe(vec!["myprog", "too", "many", "values"]);
/// assert!(result.is_err());
/// assert_eq!(result.unwrap_err().kind, ErrorKind::TooManyValues);
2015-09-05 22:12:46 +00:00
/// ```
/// [`Arg::max_values`]: ./struct.Arg.html#method.max_values
TooManyValues,
/// Occurs when the user provides fewer values for an argument than were defined by setting
/// [`Arg::min_values`].
///
/// # Examples
///
2016-01-26 19:02:10 +00:00
/// ```rust
/// # use clap::{App, Arg, ErrorKind};
/// let result = App::new("myprog")
/// .arg(Arg::with_name("some_opt")
/// .long("opt")
/// .min_values(3))
/// .get_matches_from_safe(vec!["myprog", "--opt", "too", "few"]);
/// assert!(result.is_err());
/// assert_eq!(result.unwrap_err().kind, ErrorKind::TooFewValues);
2015-09-05 22:12:46 +00:00
/// ```
/// [`Arg::min_values`]: ./struct.Arg.html#method.min_values
TooFewValues,
/// Occurs when the user provides a different number of values for an argument than what's
/// been defined by setting [`Arg::number_of_values`] or than was implicitly set by
/// [`Arg::value_names`].
///
/// # Examples
///
2016-01-26 19:02:10 +00:00
/// ```rust
/// # use clap::{App, Arg, ErrorKind};
/// let result = App::new("myprog")
/// .arg(Arg::with_name("some_opt")
/// .long("opt")
/// .takes_value(true)
/// .number_of_values(2))
2016-01-26 19:02:10 +00:00
/// .get_matches_from_safe(vec!["myprog", "--opt", "wrong"]);
/// assert!(result.is_err());
/// assert_eq!(result.unwrap_err().kind, ErrorKind::WrongNumberOfValues);
/// ```
/// [`Arg::number_of_values`]
/// [`Arg::number_of_values`]: ./struct.Arg.html#method.number_of_values
/// [`Arg::value_names`]: ./struct.Arg.html#method.value_names
WrongNumberOfValues,
2016-01-26 19:02:10 +00:00
/// Occurs when the user provides two values which conflict with each other and can't be used
/// together.
///
/// # Examples
///
2016-01-26 19:02:10 +00:00
/// ```rust
/// # use clap::{App, Arg, ErrorKind};
/// let result = App::new("myprog")
/// .arg(Arg::with_name("debug")
/// .long("debug")
/// .conflicts_with("color"))
/// .arg(Arg::with_name("color")
/// .long("color"))
/// .get_matches_from_safe(vec!["myprog", "--debug", "--color"]);
/// assert!(result.is_err());
/// assert_eq!(result.unwrap_err().kind, ErrorKind::ArgumentConflict);
2015-09-05 22:12:46 +00:00
/// ```
ArgumentConflict,
2016-01-29 02:51:50 +00:00
/// Occurs when the user does not provide one or more required arguments.
///
/// # Examples
///
2016-01-26 19:02:10 +00:00
/// ```rust
/// # use clap::{App, Arg, ErrorKind};
/// let result = App::new("myprog")
/// .arg(Arg::with_name("debug")
/// .required(true))
/// .get_matches_from_safe(vec!["myprog"]);
/// assert!(result.is_err());
/// assert_eq!(result.unwrap_err().kind, ErrorKind::MissingRequiredArgument);
2015-09-05 22:12:46 +00:00
/// ```
MissingRequiredArgument,
/// Occurs when a subcommand is required (as defined by [`AppSettings::SubcommandRequired`]),
/// but the user does not provide one.
///
/// # Examples
///
2016-01-26 19:02:10 +00:00
/// ```rust
/// # use clap::{App, AppSettings, SubCommand, ErrorKind};
/// let err = App::new("myprog")
/// .setting(AppSettings::SubcommandRequired)
2016-01-26 19:02:10 +00:00
/// .subcommand(SubCommand::with_name("test"))
/// .get_matches_from_safe(vec![
/// "myprog",
/// ]);
/// assert!(err.is_err());
/// assert_eq!(err.unwrap_err().kind, ErrorKind::MissingSubcommand);
/// # ;
2015-09-05 22:12:46 +00:00
/// ```
/// [`AppSettings::SubcommandRequired`]: ./enum.AppSettings.html#variant.SubcommandRequired
MissingSubcommand,
/// Occurs when either an argument or [`SubCommand`] is required, as defined by
/// [`AppSettings::ArgRequiredElseHelp`] but the user did not provide one.
///
/// # Examples
///
2016-01-26 19:02:10 +00:00
/// ```rust
/// # use clap::{App, Arg, AppSettings, ErrorKind, SubCommand};
/// let result = App::new("myprog")
/// .setting(AppSettings::ArgRequiredElseHelp)
2015-11-01 14:02:37 +00:00
/// .subcommand(SubCommand::with_name("config")
/// .about("Used for configuration")
/// .arg(Arg::with_name("config_file")
/// .help("The configuration file to use")))
/// .get_matches_from_safe(vec!["myprog"]);
/// assert!(result.is_err());
/// assert_eq!(result.unwrap_err().kind, ErrorKind::MissingArgumentOrSubcommand);
/// ```
/// [`SubCommand`]: ./struct.SubCommand.html
/// [`AppSettings::ArgRequiredElseHelp`]: ./enum.AppSettings.html#variant.ArgRequiredElseHelp
MissingArgumentOrSubcommand,
/// Occurs when the user provides an argument multiple times which has not been set to allow
/// multiple uses.
///
/// # Examples
///
2016-01-26 19:02:10 +00:00
/// ```rust
/// # use clap::{App, Arg, ErrorKind};
/// let result = App::new("myprog")
/// .arg(Arg::with_name("debug")
2016-01-26 19:02:10 +00:00
/// .long("debug")
/// .multiple(false))
/// .get_matches_from_safe(vec!["myprog", "--debug", "--debug"]);
/// assert!(result.is_err());
/// assert_eq!(result.unwrap_err().kind, ErrorKind::UnexpectedMultipleUsage);
2015-09-05 22:12:46 +00:00
/// ```
2015-09-07 01:07:46 +00:00
UnexpectedMultipleUsage,
/// Occurs when the user provides a value containing invalid UTF-8 for an argument and
/// [`AppSettings::StrictUtf8`] is set.
///
/// # Platform Speicific
///
/// Non-Windows platforms only (such as Linux, Unix, OSX, etc.)
///
/// # Examples
///
/// ```ignore
/// # use clap::{App, Arg, ErrorKind, AppSettings};
/// # use std::os::unix::ffi::OsStringExt;
/// # use std::ffi::OsString;
/// let result = App::new("myprog")
/// .setting(AppSettings::StrictUtf8)
/// .arg(Arg::with_name("utf8")
/// .short("u")
/// .takes_value(true))
/// .get_matches_from_safe(vec![OsString::from("myprog"),
/// OsString::from("-u")
/// OsString::from_vec(vec![0xE9])]);
/// assert!(result.is_err());
/// assert_eq!(result.unwrap_err().kind, ErrorKind::InvalidUtf8);
/// ```
/// [`AppSettings::StrictUtf8`]: ./enum.AppSettings.html#variant.StrictUtf8
InvalidUtf8,
/// Not a true "error" as it means `--help` or similar was used. The help message will be sent
/// to `stdout`.
///
/// **Note**: If the help is displayed due to an error (such as missing subcommands) it will
2016-01-29 02:51:50 +00:00
/// be sent to `stderr` instead of `stdout`.
///
/// # Examples
///
2016-01-26 19:02:10 +00:00
/// ```rust
/// # use clap::{App, Arg, ErrorKind};
/// let result = App::new("myprog")
/// .get_matches_from_safe(vec!["myprog", "--help"]);
/// assert!(result.is_err());
/// assert_eq!(result.unwrap_err().kind, ErrorKind::HelpDisplayed);
/// ```
HelpDisplayed,
/// Not a true "error" as it means `--version` or similar was used. The message will be sent
2016-01-29 02:51:50 +00:00
/// to `stdout`.
///
/// # Examples
///
2016-01-26 19:02:10 +00:00
/// ```rust
/// # use clap::{App, Arg, ErrorKind};
/// let result = App::new("myprog")
/// .get_matches_from_safe(vec!["myprog", "--version"]);
/// assert!(result.is_err());
/// assert_eq!(result.unwrap_err().kind, ErrorKind::VersionDisplayed);
/// ```
2015-10-28 14:23:59 +00:00
VersionDisplayed,
/// Occurs when using the [`value_t!`] and [`values_t!`] macros to convert an argument value
/// into type `T`, but the argument you requested wasn't used. I.e. you asked for an argument
/// with name `config` to be converted, but `config` wasn't used by the user.
/// [`value_t!`]: ./macro.value_t!.html
/// [`values_t!`]: ./macro.values_t!.html
ArgumentNotFound,
/// Represents an [I/O error], typically while writing to `stderr` or `stdout`.
/// [I/O error]: https://doc.rust-lang.org/std/io/struct.Error.html
2015-10-31 05:19:24 +00:00
Io,
/// Represents an Rust's [Format error] as part of [`Display`] , typically while writing to
/// `stderr` or `stdout`.
/// [`Display`]: https://doc.rust-lang.org/std/fmt/trait.Display.html
/// [Format error]: https://doc.rust-lang.org/std/fmt/struct.Error.html
Format,
}
2016-01-29 02:51:50 +00:00
/// Command Line Argument Parser Error
#[derive(Debug)]
pub struct Error {
/// Formated error message
pub message: String,
2015-10-31 05:19:24 +00:00
/// The type of error
pub kind: ErrorKind,
/// Any additional information passed along, such as the argument name that caused the error
pub info: Option<Vec<String>>,
}
impl Error {
2015-10-31 05:19:24 +00:00
/// Should the message be written to `stdout` or not
pub fn use_stderr(&self) -> bool {
match self.kind {
2016-05-06 21:35:53 +00:00
ErrorKind::HelpDisplayed |
ErrorKind::VersionDisplayed => false,
2015-11-09 07:22:12 +00:00
_ => true,
2015-10-31 05:19:24 +00:00
}
}
/// Prints the error to `stderr` and exits with a status of `1`
pub fn exit(&self) -> ! {
2015-10-31 05:19:24 +00:00
if self.use_stderr() {
wlnerr!("{}", self.message);
process::exit(1);
}
let out = io::stdout();
writeln!(&mut out.lock(), "{}", self.message).expect("Error writing Error to stdout");
process::exit(0);
}
#[doc(hidden)]
pub fn write_to<W: Write>(&self, w: &mut W) -> io::Result<()> {
write!(w, "{}", self.message)
}
#[doc(hidden)]
pub fn argument_conflict<'a, 'b, A, O, U>(arg: &A, other: Option<O>, usage: U, color: fmt::ColorWhen) -> Self
where A: AnyArg<'a, 'b> + Display,
O: Into<String>,
U: Display
{
let mut v = vec![arg.name().to_owned()];
let c = fmt::Colorizer {
use_stderr: true,
when: color
};
Error {
message: format!("{} The argument '{}' cannot be used with {}\n\n\
{}\n\n\
For more information try {}",
c.error("error:"),
c.warning(&*arg.to_string()),
2016-05-06 21:35:53 +00:00
match other {
Some(name) => {
let n = name.into();
v.push(n.clone());
c.warning(format!("'{}'", n))
2016-05-06 21:35:53 +00:00
}
None => c.none("one or more of the other specified arguments".to_owned()),
2016-05-06 21:35:53 +00:00
},
usage,
c.good("--help")),
kind: ErrorKind::ArgumentConflict,
info: Some(v)
}
}
#[doc(hidden)]
pub fn empty_value<'a, 'b, A, U>(arg: &A, usage: U, color: fmt::ColorWhen) -> Self
where A: AnyArg<'a, 'b> + Display,
U: Display
{
let c = fmt::Colorizer {
use_stderr: true,
when: color
};
Error {
message: format!("{} The argument '{}' requires a value but none was supplied\
\n\n\
{}\n\n\
For more information try {}",
c.error("error:"),
c.warning(arg.to_string()),
2016-05-06 21:35:53 +00:00
usage,
c.good("--help")),
kind: ErrorKind::EmptyValue,
info: Some(vec![arg.name().to_owned()]),
}
}
#[doc(hidden)]
pub fn invalid_value<'a, 'b, B, G, A, U>(bad_val: B, good_vals: &[G], arg: &A, usage: U, color: fmt::ColorWhen) -> Self
where B: AsRef<str>,
G: AsRef<str> + Display,
A: AnyArg<'a, 'b> + Display,
U: Display
{
2016-07-05 20:25:35 +00:00
let c = fmt::Colorizer {
use_stderr: true,
when: color
};
2016-05-06 21:35:53 +00:00
let suffix =
suggestions::did_you_mean_suffix(bad_val.as_ref(),
good_vals.iter(),
suggestions::DidYouMeanMessageStyle::EnumValue);
let mut sorted = vec![];
for v in good_vals {
2016-07-05 20:25:35 +00:00
let val = format!("{}", c.good(v));
sorted.push(val);
}
sorted.sort();
2016-07-05 20:04:00 +00:00
let valid_values = sorted.join(", ");
Error {
message: format!("{} '{}' isn't a valid value for '{}'\n\t\
2016-07-05 20:04:00 +00:00
[values: {}]\n\
{}\n\n\
{}\n\n\
For more information try {}",
c.error("error:"),
c.warning(bad_val.as_ref()),
c.warning(arg.to_string()),
2016-05-06 21:35:53 +00:00
valid_values,
suffix.0,
usage,
c.good("--help")),
kind: ErrorKind::InvalidValue,
info: Some(vec![arg.name().to_owned(), bad_val.as_ref().to_owned()]),
}
}
#[doc(hidden)]
pub fn invalid_subcommand<S, D, N, U>(subcmd: S, did_you_mean: D, name: N, usage: U, color: fmt::ColorWhen) -> Self
where S: Into<String>,
D: AsRef<str> + Display,
N: Display,
U: Display
{
let s = subcmd.into();
let c = fmt::Colorizer {
use_stderr: true,
when: color
};
Error {
message: format!("{} The subcommand '{}' wasn't recognized\n\t\
2016-07-05 20:01:04 +00:00
Did you mean '{}'?\n\n\
If you believe you received this message in error, try \
re-running with '{} {} {}'\n\n\
{}\n\n\
For more information try {}",
c.error("error:"),
c.warning(&*s),
c.good(did_you_mean.as_ref()),
2016-05-06 21:35:53 +00:00
name,
c.good("--"),
2016-05-06 21:35:53 +00:00
&*s,
usage,
c.good("--help")),
kind: ErrorKind::InvalidSubcommand,
info: Some(vec![s]),
}
}
#[doc(hidden)]
pub fn unrecognized_subcommand<S, N>(subcmd: S, name: N, color: fmt::ColorWhen) -> Self
where S: Into<String>,
N: Display
{
let s = subcmd.into();
let c = fmt::Colorizer {
use_stderr: true,
when: color
};
Error {
message: format!("{} The subcommand '{}' wasn't recognized\n\n\
2016-07-05 20:25:35 +00:00
{}\n\t\
{} help <subcommands>...\n\n\
For more information try {}",
c.error("error:"),
c.warning(&*s),
2016-07-05 20:25:35 +00:00
c.warning("USAGE:"),
2016-05-06 21:35:53 +00:00
name,
c.good("--help")),
kind: ErrorKind::UnrecognizedSubcommand,
info: Some(vec![s]),
}
}
#[doc(hidden)]
pub fn missing_required_argument<R, U>(required: R, usage: U, color: fmt::ColorWhen) -> Self
where R: Display,
U: Display
{
let c = fmt::Colorizer {
use_stderr: true,
when: color
};
Error {
message: format!("{} The following required arguments were not provided:{}\n\n\
{}\n\n\
For more information try {}",
c.error("error:"),
2016-05-06 21:35:53 +00:00
required,
usage,
c.good("--help")),
kind: ErrorKind::MissingRequiredArgument,
info: None,
}
}
#[doc(hidden)]
pub fn missing_subcommand<N, U>(name: N, usage: U, color: fmt::ColorWhen) -> Self
where N: AsRef<str> + Display,
U: Display
{
let c = fmt::Colorizer {
use_stderr: true,
when: color
};
Error {
message: format!("{} '{}' requires a subcommand, but one was not provided\n\n\
{}\n\n\
For more information try {}",
c.error("error:"),
c.warning(name),
2016-05-06 21:35:53 +00:00
usage,
c.good("--help")),
kind: ErrorKind::MissingSubcommand,
info: None,
}
}
#[doc(hidden)]
pub fn invalid_utf8<U>(usage: U, color: fmt::ColorWhen) -> Self
where U: Display
{
let c = fmt::Colorizer {
use_stderr: true,
when: color
};
Error {
message: format!("{} Invalid UTF-8 was detected in one or more arguments\n\n\
{}\n\n\
For more information try {}",
c.error("error:"),
2016-05-06 21:35:53 +00:00
usage,
c.good("--help")),
kind: ErrorKind::InvalidUtf8,
info: None,
}
}
#[doc(hidden)]
pub fn too_many_values<'a, 'b, V, A, U>(val: V, arg: &A, usage: U, color: fmt::ColorWhen) -> Self
where V: AsRef<str> + Display + ToOwned,
A: AnyArg<'a, 'b> + Display,
U: Display
{
let v = val.as_ref();
let c = fmt::Colorizer {
use_stderr: true,
when: color
};
Error {
message: format!("{} The value '{}' was provided to '{}', but it wasn't expecting \
any more values\n\n\
{}\n\n\
For more information try {}",
c.error("error:"),
c.warning(v),
c.warning(arg.to_string()),
2016-05-06 21:35:53 +00:00
usage,
c.good("--help")),
kind: ErrorKind::TooManyValues,
info: Some(vec![arg.name().to_owned(), v.to_owned()]),
}
}
#[doc(hidden)]
pub fn too_few_values<'a, 'b, A, U>(arg: &A, min_vals: u64, curr_vals: usize, usage: U, color: fmt::ColorWhen) -> Self
where A: AnyArg<'a, 'b> + Display,
U: Display
{
let c = fmt::Colorizer {
use_stderr: true,
when: color
};
Error {
message: format!("{} The argument '{}' requires at least {} values, but only {} w{} \
provided\n\n\
{}\n\n\
For more information try {}",
c.error("error:"),
c.warning(arg.to_string()),
c.warning(min_vals.to_string()),
c.warning(curr_vals.to_string()),
2016-05-06 21:35:53 +00:00
if curr_vals > 1 {
"ere"
} else {
"as"
},
usage,
c.good("--help")),
kind: ErrorKind::TooFewValues,
info: Some(vec![arg.name().to_owned()]),
}
}
#[doc(hidden)]
pub fn value_validation(err: String, color: fmt::ColorWhen) -> Self {
let c = fmt::Colorizer {
use_stderr: true,
when: color
};
Error {
message: format!("{} {}", c.error("error:"), err),
kind: ErrorKind::ValueValidation,
info: None,
}
}
#[doc(hidden)]
pub fn value_validation_auto(err: String) -> Self {
Error::value_validation(err, fmt::ColorWhen::Auto)
}
#[doc(hidden)]
2016-05-06 21:35:53 +00:00
pub fn wrong_number_of_values<'a, 'b, A, S, U>(arg: &A,
num_vals: u64,
curr_vals: usize,
suffix: S,
usage: U, color: fmt::ColorWhen)
2016-05-06 21:35:53 +00:00
-> Self
where A: AnyArg<'a, 'b> + Display,
S: Display,
U: Display
{
let c = fmt::Colorizer {
use_stderr: true,
when: color
};
Error {
message: format!("{} The argument '{}' requires {} values, but {} w{} \
provided\n\n\
{}\n\n\
For more information try {}",
c.error("error:"),
c.warning(arg.to_string()),
c.warning(num_vals.to_string()),
c.warning(curr_vals.to_string()),
2016-05-06 21:35:53 +00:00
suffix,
usage,
c.good("--help")),
kind: ErrorKind::WrongNumberOfValues,
info: Some(vec![arg.name().to_owned()]),
}
}
#[doc(hidden)]
pub fn unexpected_multiple_usage<'a, 'b, A, U>(arg: &A, usage: U, color: fmt::ColorWhen) -> Self
where A: AnyArg<'a, 'b> + Display,
U: Display
{
let c = fmt::Colorizer {
use_stderr: true,
when: color
};
Error {
message: format!("{} The argument '{}' was provided more than once, but cannot \
be used multiple times\n\n\
{}\n\n\
For more information try {}",
c.error("error:"),
c.warning(arg.to_string()),
2016-05-06 21:35:53 +00:00
usage,
c.good("--help")),
kind: ErrorKind::UnexpectedMultipleUsage,
info: Some(vec![arg.name().to_owned()]),
}
}
#[doc(hidden)]
pub fn unknown_argument<A, U>(arg: A, did_you_mean: &str, usage: U, color: fmt::ColorWhen) -> Self
where A: Into<String>,
U: Display
{
let a = arg.into();
let c = fmt::Colorizer {
use_stderr: true,
when: color
};
Error {
2016-05-06 21:35:53 +00:00
message: format!("{} Found argument '{}' which wasn't expected, or isn't valid in \
this context{}\n\
{}\n\n\
For more information try {}",
c.error("error:"),
c.warning(&*a),
2016-05-06 21:35:53 +00:00
if did_you_mean.is_empty() {
"\n".to_owned()
} else {
format!("{}\n", did_you_mean)
},
usage,
c.good("--help")),
kind: ErrorKind::UnknownArgument,
info: Some(vec![a]),
}
}
#[doc(hidden)]
pub fn io_error(e: &Error, color: fmt::ColorWhen) -> Self {
let c = fmt::Colorizer {
use_stderr: true,
when: color
};
Error {
message: format!("{} {}", c.error("error:"), e.description()),
kind: ErrorKind::Io,
info: None,
}
}
#[doc(hidden)]
pub fn argument_not_found_auto<A>(arg: A) -> Self
2016-05-06 21:35:53 +00:00
where A: Into<String>
{
let a = arg.into();
let c = fmt::Colorizer {
use_stderr: true,
when: fmt::ColorWhen::Auto
};
Error {
2016-05-06 21:35:53 +00:00
message: format!("{} The argument '{}' wasn't found",
c.error("error:"),
2016-05-06 21:35:53 +00:00
a.clone()),
kind: ErrorKind::ArgumentNotFound,
info: Some(vec![a]),
}
}
/// Create an error with a custom description.
///
/// This can be used in combination with `Error::exit` to exit your program
/// with a custom error message.
pub fn with_description(description: &str, kind: ErrorKind) -> Self {
let c = fmt::Colorizer {
use_stderr: true,
when: fmt::ColorWhen::Auto
};
Error {
message: format!("{} {}", c.error("error:"), description),
kind: kind,
info: None,
}
}
}
impl StdError for Error {
fn description(&self) -> &str {
&*self.message
}
}
impl Display for Error {
fn fmt(&self, f: &mut std_fmt::Formatter) -> std_fmt::Result {
writeln!(f, "{}", self.message)
}
2015-09-07 01:07:46 +00:00
}
2015-10-31 05:19:24 +00:00
impl From<io::Error> for Error {
2015-10-31 05:19:24 +00:00
fn from(e: io::Error) -> Self {
Error::with_description(e.description(), ErrorKind::Io)
2015-10-31 05:19:24 +00:00
}
}
impl From<std_fmt::Error> for Error {
fn from(e: std_fmt::Error) -> Self {
Error::with_description(e.description(), ErrorKind::Format)
}
}