Merge pull request #123 from murarth/ratio-parse-fix
Check for zero denominator when parsing Ratio
This commit is contained in:
commit
9e5a209dcd
|
@ -404,15 +404,18 @@ impl<T: Clone + Integer + PartialOrd> Num for Ratio<T> {
|
|||
fn from_str_radix(s: &str, radix: u32) -> Result<Ratio<T>, ParseRatioError> {
|
||||
let split: Vec<&str> = s.splitn(2, '/').collect();
|
||||
if split.len() < 2 {
|
||||
Err(ParseRatioError)
|
||||
Err(ParseRatioError{kind: RatioErrorKind::ParseError})
|
||||
} else {
|
||||
let a_result: Result<T, _> = T::from_str_radix(
|
||||
split[0],
|
||||
radix).map_err(|_| ParseRatioError);
|
||||
radix).map_err(|_| ParseRatioError{kind: RatioErrorKind::ParseError});
|
||||
a_result.and_then(|a| {
|
||||
let b_result: Result<T, _> =
|
||||
T::from_str_radix(split[1], radix).map_err(|_| ParseRatioError);
|
||||
b_result.and_then(|b| {
|
||||
T::from_str_radix(split[1], radix).map_err(
|
||||
|_| ParseRatioError{kind: RatioErrorKind::ParseError});
|
||||
b_result.and_then(|b| if b.is_zero() {
|
||||
Err(ParseRatioError{kind: RatioErrorKind::ZeroDenominator})
|
||||
} else {
|
||||
Ok(Ratio::new(a.clone(), b.clone()))
|
||||
})
|
||||
})
|
||||
|
@ -470,28 +473,50 @@ impl<T: FromStr + Clone + Integer + PartialOrd> FromStr for Ratio<T> {
|
|||
fn from_str(s: &str) -> Result<Ratio<T>, ParseRatioError> {
|
||||
let mut split = s.splitn(2, '/');
|
||||
|
||||
let n = try!(split.next().ok_or(ParseRatioError));
|
||||
let num = try!(FromStr::from_str(n).map_err(|_| ParseRatioError));
|
||||
let n = try!(split.next().ok_or(
|
||||
ParseRatioError{kind: RatioErrorKind::ParseError}));
|
||||
let num = try!(FromStr::from_str(n).map_err(
|
||||
|_| ParseRatioError{kind: RatioErrorKind::ParseError}));
|
||||
|
||||
let d = split.next().unwrap_or("1");
|
||||
let den = try!(FromStr::from_str(d).map_err(|_| ParseRatioError));
|
||||
let den = try!(FromStr::from_str(d).map_err(
|
||||
|_| ParseRatioError{kind: RatioErrorKind::ParseError}));
|
||||
|
||||
if Zero::is_zero(&den) {
|
||||
Err(ParseRatioError{kind: RatioErrorKind::ZeroDenominator})
|
||||
} else {
|
||||
Ok(Ratio::new(num, den))
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// FIXME: Bubble up specific errors
|
||||
#[derive(Copy, Clone, Debug, PartialEq)]
|
||||
pub struct ParseRatioError;
|
||||
pub struct ParseRatioError { kind: RatioErrorKind }
|
||||
|
||||
#[derive(Copy, Clone, Debug, PartialEq)]
|
||||
enum RatioErrorKind {
|
||||
ParseError,
|
||||
ZeroDenominator,
|
||||
}
|
||||
|
||||
impl fmt::Display for ParseRatioError {
|
||||
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||
"failed to parse provided string".fmt(f)
|
||||
self.description().fmt(f)
|
||||
}
|
||||
}
|
||||
|
||||
impl Error for ParseRatioError {
|
||||
fn description(&self) -> &str { "failed to parse bigint/biguint" }
|
||||
fn description(&self) -> &str { self.kind.description() }
|
||||
}
|
||||
|
||||
impl RatioErrorKind {
|
||||
fn description(&self) -> &'static str {
|
||||
match *self {
|
||||
RatioErrorKind::ParseError => "failed to parse integer",
|
||||
RatioErrorKind::ZeroDenominator => "zero value denominator",
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
#[cfg(test)]
|
||||
|
@ -817,7 +842,7 @@ mod test {
|
|||
assert!(rational.is_err());
|
||||
}
|
||||
|
||||
let xs = ["0 /1", "abc", "", "1/", "--1/2","3/2/1"];
|
||||
let xs = ["0 /1", "abc", "", "1/", "--1/2","3/2/1", "1/0"];
|
||||
for &s in xs.iter() {
|
||||
test(s);
|
||||
}
|
||||
|
|
Loading…
Reference in New Issue