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> {
|
fn from_str_radix(s: &str, radix: u32) -> Result<Ratio<T>, ParseRatioError> {
|
||||||
let split: Vec<&str> = s.splitn(2, '/').collect();
|
let split: Vec<&str> = s.splitn(2, '/').collect();
|
||||||
if split.len() < 2 {
|
if split.len() < 2 {
|
||||||
Err(ParseRatioError)
|
Err(ParseRatioError{kind: RatioErrorKind::ParseError})
|
||||||
} else {
|
} else {
|
||||||
let a_result: Result<T, _> = T::from_str_radix(
|
let a_result: Result<T, _> = T::from_str_radix(
|
||||||
split[0],
|
split[0],
|
||||||
radix).map_err(|_| ParseRatioError);
|
radix).map_err(|_| ParseRatioError{kind: RatioErrorKind::ParseError});
|
||||||
a_result.and_then(|a| {
|
a_result.and_then(|a| {
|
||||||
let b_result: Result<T, _> =
|
let b_result: Result<T, _> =
|
||||||
T::from_str_radix(split[1], radix).map_err(|_| ParseRatioError);
|
T::from_str_radix(split[1], radix).map_err(
|
||||||
b_result.and_then(|b| {
|
|_| 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()))
|
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> {
|
fn from_str(s: &str) -> Result<Ratio<T>, ParseRatioError> {
|
||||||
let mut split = s.splitn(2, '/');
|
let mut split = s.splitn(2, '/');
|
||||||
|
|
||||||
let n = try!(split.next().ok_or(ParseRatioError));
|
let n = try!(split.next().ok_or(
|
||||||
let num = try!(FromStr::from_str(n).map_err(|_| ParseRatioError));
|
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 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))
|
Ok(Ratio::new(num, den))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// FIXME: Bubble up specific errors
|
// FIXME: Bubble up specific errors
|
||||||
#[derive(Copy, Clone, Debug, PartialEq)]
|
#[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 {
|
impl fmt::Display for ParseRatioError {
|
||||||
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||||
"failed to parse provided string".fmt(f)
|
self.description().fmt(f)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Error for ParseRatioError {
|
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)]
|
#[cfg(test)]
|
||||||
|
@ -817,7 +842,7 @@ mod test {
|
||||||
assert!(rational.is_err());
|
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() {
|
for &s in xs.iter() {
|
||||||
test(s);
|
test(s);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue