use nom8::branch::alt;
use nom8::character::digit1 as digit;
use nom8::combinator::opt;
use nom8::prelude::*;
use nom8::sequence::delimited;
use nom8::IResult;
use std::str;
use std::str::FromStr;
fn unsigned_float(i: &[u8]) -> IResult<&[u8], f32> {
let float_bytes = alt((
delimited(digit, ".", opt(digit)),
delimited(opt(digit), ".", digit),
))
.recognize();
let float_str = float_bytes.map_res(str::from_utf8);
float_str.map_res(FromStr::from_str).parse(i)
}
fn float(i: &[u8]) -> IResult<&[u8], f32> {
(opt(alt(("+", "-"))), unsigned_float)
.map(|(sign, value)| {
sign
.and_then(|s| if s[0] == b'-' { Some(-1f32) } else { None })
.unwrap_or(1f32)
* value
})
.parse(i)
}
#[test]
fn unsigned_float_test() {
assert_eq!(unsigned_float(&b"123.456;"[..]), Ok((&b";"[..], 123.456)));
assert_eq!(unsigned_float(&b"0.123;"[..]), Ok((&b";"[..], 0.123)));
assert_eq!(unsigned_float(&b"123.0;"[..]), Ok((&b";"[..], 123.0)));
assert_eq!(unsigned_float(&b"123.;"[..]), Ok((&b";"[..], 123.0)));
assert_eq!(unsigned_float(&b".123;"[..]), Ok((&b";"[..], 0.123)));
}
#[test]
fn float_test() {
assert_eq!(float(&b"123.456;"[..]), Ok((&b";"[..], 123.456)));
assert_eq!(float(&b"+123.456;"[..]), Ok((&b";"[..], 123.456)));
assert_eq!(float(&b"-123.456;"[..]), Ok((&b";"[..], -123.456)));
}