diff --git a/clap_lex/src/lib.rs b/clap_lex/src/lib.rs index d519a10c5621..edfdee6cc9b4 100644 --- a/clap_lex/src/lib.rs +++ b/clap_lex/src/lib.rs @@ -302,9 +302,7 @@ impl<'s> ParsedArg<'s> { /// Does the argument look like a number pub fn is_number(&self) -> bool { - self.to_value() - .map(|s| s.parse::().is_ok()) - .unwrap_or_default() + self.to_value().map(looks_like_number).unwrap_or_default() } /// Treat as a long-flag @@ -409,7 +407,7 @@ impl<'s> ShortFlags<'s> { /// /// Ideally call this before doing any iterator pub fn is_number(&self) -> bool { - self.invalid_suffix.is_none() && self.utf8_prefix.as_str().parse::().is_ok() + self.invalid_suffix.is_none() && looks_like_number(self.utf8_prefix.as_str()) } /// Advance the iterator, returning the next short flag on success @@ -466,3 +464,20 @@ fn split_nonutf8_once(b: &OsStr) -> (&str, Option<&OsStr>) { } } } + +fn looks_like_number(arg: &str) -> bool { + // strip a leading `-` in case this is a negative number + let arg = arg.strip_prefix("-").unwrap_or(arg); + + // Return true if this looks like an integer or a float where it's all + // digits plus an optional single dot after some digits. + let mut seen_dot = false; + for (i, c) in arg.as_bytes().iter().enumerate() { + match c { + b'0'..=b'9' => {} + b'.' if !seen_dot && i > 0 => seen_dot = true, + _ => return false, + } + } + true +}