use super::*;
mod complete {
use super::*;
use crate::branch::alt;
use crate::combinator::opt;
use crate::error::ErrorKind;
use crate::input::ParseTo;
use crate::Err;
use proptest::prelude::*;
macro_rules! assert_parse(
($left: expr, $right: expr) => {
let res: $crate::IResult<_, _, (_, ErrorKind)> = $left;
assert_eq!(res, $right);
};
);
#[test]
fn character() {
let empty: &[u8] = b"";
let a: &[u8] = b"abcd";
let b: &[u8] = b"1234";
let c: &[u8] = b"a123";
let d: &[u8] = "azé12".as_bytes();
let e: &[u8] = b" ";
let f: &[u8] = b" ;";
assert_parse!(alpha1(a), Ok((empty, a)));
assert_eq!(alpha1(b), Err(Err::Error((b, ErrorKind::Alpha))));
assert_eq!(
alpha1::<_, (_, ErrorKind), false>(c),
Ok((&c[1..], &b"a"[..]))
);
assert_eq!(
alpha1::<_, (_, ErrorKind), false>(d),
Ok(("é12".as_bytes(), &b"az"[..]))
);
assert_eq!(digit1(a), Err(Err::Error((a, ErrorKind::Digit))));
assert_eq!(digit1::<_, (_, ErrorKind), false>(b), Ok((empty, b)));
assert_eq!(digit1(c), Err(Err::Error((c, ErrorKind::Digit))));
assert_eq!(digit1(d), Err(Err::Error((d, ErrorKind::Digit))));
assert_eq!(hex_digit1::<_, (_, ErrorKind), false>(a), Ok((empty, a)));
assert_eq!(hex_digit1::<_, (_, ErrorKind), false>(b), Ok((empty, b)));
assert_eq!(hex_digit1::<_, (_, ErrorKind), false>(c), Ok((empty, c)));
assert_eq!(
hex_digit1::<_, (_, ErrorKind), false>(d),
Ok(("zé12".as_bytes(), &b"a"[..]))
);
assert_eq!(hex_digit1(e), Err(Err::Error((e, ErrorKind::HexDigit))));
assert_eq!(oct_digit1(a), Err(Err::Error((a, ErrorKind::OctDigit))));
assert_eq!(oct_digit1::<_, (_, ErrorKind), false>(b), Ok((empty, b)));
assert_eq!(oct_digit1(c), Err(Err::Error((c, ErrorKind::OctDigit))));
assert_eq!(oct_digit1(d), Err(Err::Error((d, ErrorKind::OctDigit))));
assert_eq!(alphanumeric1::<_, (_, ErrorKind), false>(a), Ok((empty, a)));
assert_eq!(alphanumeric1::<_, (_, ErrorKind), false>(c), Ok((empty, c)));
assert_eq!(
alphanumeric1::<_, (_, ErrorKind), false>(d),
Ok(("é12".as_bytes(), &b"az"[..]))
);
assert_eq!(space1::<_, (_, ErrorKind), false>(e), Ok((empty, e)));
assert_eq!(
space1::<_, (_, ErrorKind), false>(f),
Ok((&b";"[..], &b" "[..]))
);
}
#[cfg(feature = "alloc")]
#[test]
fn character_s() {
let empty = "";
let a = "abcd";
let b = "1234";
let c = "a123";
let d = "azé12";
let e = " ";
assert_eq!(alpha1::<_, (_, ErrorKind), false>(a), Ok((empty, a)));
assert_eq!(alpha1(b), Err(Err::Error((b, ErrorKind::Alpha))));
assert_eq!(
alpha1::<_, (_, ErrorKind), false>(c),
Ok((&c[1..], &"a"[..]))
);
assert_eq!(
alpha1::<_, (_, ErrorKind), false>(d),
Ok(("é12", &"az"[..]))
);
assert_eq!(digit1(a), Err(Err::Error((a, ErrorKind::Digit))));
assert_eq!(digit1::<_, (_, ErrorKind), false>(b), Ok((empty, b)));
assert_eq!(digit1(c), Err(Err::Error((c, ErrorKind::Digit))));
assert_eq!(digit1(d), Err(Err::Error((d, ErrorKind::Digit))));
assert_eq!(hex_digit1::<_, (_, ErrorKind), false>(a), Ok((empty, a)));
assert_eq!(hex_digit1::<_, (_, ErrorKind), false>(b), Ok((empty, b)));
assert_eq!(hex_digit1::<_, (_, ErrorKind), false>(c), Ok((empty, c)));
assert_eq!(
hex_digit1::<_, (_, ErrorKind), false>(d),
Ok(("zé12", &"a"[..]))
);
assert_eq!(hex_digit1(e), Err(Err::Error((e, ErrorKind::HexDigit))));
assert_eq!(oct_digit1(a), Err(Err::Error((a, ErrorKind::OctDigit))));
assert_eq!(oct_digit1::<_, (_, ErrorKind), false>(b), Ok((empty, b)));
assert_eq!(oct_digit1(c), Err(Err::Error((c, ErrorKind::OctDigit))));
assert_eq!(oct_digit1(d), Err(Err::Error((d, ErrorKind::OctDigit))));
assert_eq!(alphanumeric1::<_, (_, ErrorKind), false>(a), Ok((empty, a)));
assert_eq!(alphanumeric1::<_, (_, ErrorKind), false>(c), Ok((empty, c)));
assert_eq!(
alphanumeric1::<_, (_, ErrorKind), false>(d),
Ok(("é12", "az"))
);
assert_eq!(space1::<_, (_, ErrorKind), false>(e), Ok((empty, e)));
}
use crate::input::Offset;
#[test]
fn offset() {
let a = &b"abcd;"[..];
let b = &b"1234;"[..];
let c = &b"a123;"[..];
let d = &b" \t;"[..];
let e = &b" \t\r\n;"[..];
let f = &b"123abcDEF;"[..];
match alpha1::<_, (_, ErrorKind), false>(a) {
Ok((i, _)) => {
assert_eq!(a.offset(i) + i.len(), a.len());
}
_ => panic!("wrong return type in offset test for alpha"),
}
match digit1::<_, (_, ErrorKind), false>(b) {
Ok((i, _)) => {
assert_eq!(b.offset(i) + i.len(), b.len());
}
_ => panic!("wrong return type in offset test for digit"),
}
match alphanumeric1::<_, (_, ErrorKind), false>(c) {
Ok((i, _)) => {
assert_eq!(c.offset(i) + i.len(), c.len());
}
_ => panic!("wrong return type in offset test for alphanumeric"),
}
match space1::<_, (_, ErrorKind), false>(d) {
Ok((i, _)) => {
assert_eq!(d.offset(i) + i.len(), d.len());
}
_ => panic!("wrong return type in offset test for space"),
}
match multispace1::<_, (_, ErrorKind), false>(e) {
Ok((i, _)) => {
assert_eq!(e.offset(i) + i.len(), e.len());
}
_ => panic!("wrong return type in offset test for multispace"),
}
match hex_digit1::<_, (_, ErrorKind), false>(f) {
Ok((i, _)) => {
assert_eq!(f.offset(i) + i.len(), f.len());
}
_ => panic!("wrong return type in offset test for hex_digit"),
}
match oct_digit1::<_, (_, ErrorKind), false>(f) {
Ok((i, _)) => {
assert_eq!(f.offset(i) + i.len(), f.len());
}
_ => panic!("wrong return type in offset test for oct_digit"),
}
}
#[test]
fn is_not_line_ending_bytes() {
let a: &[u8] = b"ab12cd\nefgh";
assert_eq!(
not_line_ending::<_, (_, ErrorKind), false>(a),
Ok((&b"\nefgh"[..], &b"ab12cd"[..]))
);
let b: &[u8] = b"ab12cd\nefgh\nijkl";
assert_eq!(
not_line_ending::<_, (_, ErrorKind), false>(b),
Ok((&b"\nefgh\nijkl"[..], &b"ab12cd"[..]))
);
let c: &[u8] = b"ab12cd\r\nefgh\nijkl";
assert_eq!(
not_line_ending::<_, (_, ErrorKind), false>(c),
Ok((&b"\r\nefgh\nijkl"[..], &b"ab12cd"[..]))
);
let d: &[u8] = b"ab12cd";
assert_eq!(
not_line_ending::<_, (_, ErrorKind), false>(d),
Ok((&[][..], &d[..]))
);
}
#[test]
fn is_not_line_ending_str() {
let f = "βèƒôřè\rÂßÇáƒĆèř";
assert_eq!(not_line_ending(f), Err(Err::Error((f, ErrorKind::Tag))));
let g2: &str = "ab12cd";
assert_eq!(
not_line_ending::<_, (_, ErrorKind), false>(g2),
Ok(("", g2))
);
}
#[test]
fn hex_digit_test() {
let i = &b"0123456789abcdefABCDEF;"[..];
assert_parse!(hex_digit1(i), Ok((&b";"[..], &i[..i.len() - 1])));
let i = &b"g"[..];
assert_parse!(
hex_digit1(i),
Err(Err::Error(error_position!(i, ErrorKind::HexDigit)))
);
let i = &b"G"[..];
assert_parse!(
hex_digit1(i),
Err(Err::Error(error_position!(i, ErrorKind::HexDigit)))
);
assert!(AsChar::is_hex_digit(b'0'));
assert!(AsChar::is_hex_digit(b'9'));
assert!(AsChar::is_hex_digit(b'a'));
assert!(AsChar::is_hex_digit(b'f'));
assert!(AsChar::is_hex_digit(b'A'));
assert!(AsChar::is_hex_digit(b'F'));
assert!(!AsChar::is_hex_digit(b'g'));
assert!(!AsChar::is_hex_digit(b'G'));
assert!(!AsChar::is_hex_digit(b'/'));
assert!(!AsChar::is_hex_digit(b':'));
assert!(!AsChar::is_hex_digit(b'@'));
assert!(!AsChar::is_hex_digit(b'\x60'));
}
#[test]
fn oct_digit_test() {
let i = &b"01234567;"[..];
assert_parse!(oct_digit1(i), Ok((&b";"[..], &i[..i.len() - 1])));
let i = &b"8"[..];
assert_parse!(
oct_digit1(i),
Err(Err::Error(error_position!(i, ErrorKind::OctDigit)))
);
assert!(AsChar::is_oct_digit(b'0'));
assert!(AsChar::is_oct_digit(b'7'));
assert!(!AsChar::is_oct_digit(b'8'));
assert!(!AsChar::is_oct_digit(b'9'));
assert!(!AsChar::is_oct_digit(b'a'));
assert!(!AsChar::is_oct_digit(b'A'));
assert!(!AsChar::is_oct_digit(b'/'));
assert!(!AsChar::is_oct_digit(b':'));
assert!(!AsChar::is_oct_digit(b'@'));
assert!(!AsChar::is_oct_digit(b'\x60'));
}
#[test]
fn full_line_windows() {
use crate::sequence::pair;
fn take_full_line(i: &[u8]) -> IResult<&[u8], (&[u8], &[u8])> {
pair(not_line_ending, line_ending)(i)
}
let input = b"abc\r\n";
let output = take_full_line(input);
assert_eq!(output, Ok((&b""[..], (&b"abc"[..], &b"\r\n"[..]))));
}
#[test]
fn full_line_unix() {
use crate::sequence::pair;
fn take_full_line(i: &[u8]) -> IResult<&[u8], (&[u8], &[u8])> {
pair(not_line_ending, line_ending)(i)
}
let input = b"abc\n";
let output = take_full_line(input);
assert_eq!(output, Ok((&b""[..], (&b"abc"[..], &b"\n"[..]))));
}
#[test]
fn check_windows_lineending() {
let input = b"\r\n";
let output = line_ending(&input[..]);
assert_parse!(output, Ok((&b""[..], &b"\r\n"[..])));
}
#[test]
fn check_unix_lineending() {
let input = b"\n";
let output = line_ending(&input[..]);
assert_parse!(output, Ok((&b""[..], &b"\n"[..])));
}
#[test]
fn cr_lf() {
assert_parse!(crlf(&b"\r\na"[..]), Ok((&b"a"[..], &b"\r\n"[..])));
assert_parse!(
crlf(&b"\r"[..]),
Err(Err::Error(error_position!(&b"\r"[..], ErrorKind::CrLf)))
);
assert_parse!(
crlf(&b"\ra"[..]),
Err(Err::Error(error_position!(&b"\ra"[..], ErrorKind::CrLf)))
);
assert_parse!(crlf("\r\na"), Ok(("a", "\r\n")));
assert_parse!(
crlf("\r"),
Err(Err::Error(error_position!(&"\r"[..], ErrorKind::CrLf)))
);
assert_parse!(
crlf("\ra"),
Err(Err::Error(error_position!("\ra", ErrorKind::CrLf)))
);
}
#[test]
fn end_of_line() {
assert_parse!(line_ending(&b"\na"[..]), Ok((&b"a"[..], &b"\n"[..])));
assert_parse!(line_ending(&b"\r\na"[..]), Ok((&b"a"[..], &b"\r\n"[..])));
assert_parse!(
line_ending(&b"\r"[..]),
Err(Err::Error(error_position!(&b"\r"[..], ErrorKind::CrLf)))
);
assert_parse!(
line_ending(&b"\ra"[..]),
Err(Err::Error(error_position!(&b"\ra"[..], ErrorKind::CrLf)))
);
assert_parse!(line_ending("\na"), Ok(("a", "\n")));
assert_parse!(line_ending("\r\na"), Ok(("a", "\r\n")));
assert_parse!(
line_ending("\r"),
Err(Err::Error(error_position!(&"\r"[..], ErrorKind::CrLf)))
);
assert_parse!(
line_ending("\ra"),
Err(Err::Error(error_position!("\ra", ErrorKind::CrLf)))
);
}
fn digit_to_i16(input: &str) -> IResult<&str, i16> {
let i = input;
let (i, opt_sign) = opt(alt(('+', '-')))(i)?;
let sign = match opt_sign {
Some('+') => true,
Some('-') => false,
_ => true,
};
let (i, s) = match digit1::<_, crate::error::Error<_>, false>(i) {
Ok((i, s)) => (i, s),
Err(_) => {
return Err(Err::Error(crate::error::Error::from_error_kind(
input,
ErrorKind::Digit,
)))
}
};
match s.parse_to() {
Some(n) => {
if sign {
Ok((i, n))
} else {
Ok((i, -n))
}
}
None => Err(Err::Error(crate::error::Error::from_error_kind(
i,
ErrorKind::Digit,
))),
}
}
fn digit_to_u32(i: &str) -> IResult<&str, u32> {
let (i, s) = digit1(i)?;
match s.parse_to() {
Some(n) => Ok((i, n)),
None => Err(Err::Error(crate::error::Error::from_error_kind(
i,
ErrorKind::Digit,
))),
}
}
proptest! {
#[test]
fn ints(s in "\\PC*") {
let res1 = digit_to_i16(&s);
let res2 = i16(s.as_str());
assert_eq!(res1, res2);
}
#[test]
fn uints(s in "\\PC*") {
let res1 = digit_to_u32(&s);
let res2 = u32(s.as_str());
assert_eq!(res1, res2);
}
}
#[test]
#[cfg(feature = "std")]
fn float_test() {
let mut test_cases = vec![
"+3.14",
"3.14",
"-3.14",
"0",
"0.0",
"1.",
".789",
"-.5",
"1e7",
"-1E-7",
".3e-2",
"1.e4",
"1.2e4",
"12.34",
"-1.234E-12",
"-1.234e-12",
"0.00000000000000000087",
];
for test in test_cases.drain(..) {
let expected32 = str::parse::<f32>(test).unwrap();
let expected64 = str::parse::<f64>(test).unwrap();
println!("now parsing: {} -> {}", test, expected32);
let larger = format!("{}", test);
assert_parse!(recognize_float(&larger[..]), Ok(("", test)));
assert_parse!(f32(larger.as_bytes()), Ok((&b""[..], expected32)));
assert_parse!(f32(&larger[..]), Ok(("", expected32)));
assert_parse!(f64(larger.as_bytes()), Ok((&b""[..], expected64)));
assert_parse!(f64(&larger[..]), Ok(("", expected64)));
}
let remaining_exponent = "-1.234E-";
assert_parse!(
recognize_float(remaining_exponent),
Err(Err::Failure(("", ErrorKind::Digit)))
);
let (_i, nan) = f32::<_, (), false>("NaN").unwrap();
assert!(nan.is_nan());
let (_i, inf) = f32::<_, (), false>("inf").unwrap();
assert!(inf.is_infinite());
let (_i, inf) = f32::<_, (), false>("infinite").unwrap();
assert!(inf.is_infinite());
}
#[cfg(feature = "std")]
fn parse_f64(i: &str) -> IResult<&str, f64, ()> {
#[allow(deprecated)] match crate::number::complete::recognize_float_or_exceptions(i) {
Err(e) => Err(e),
Ok((i, s)) => {
if s.is_empty() {
return Err(Err::Error(()));
}
match s.parse_to() {
Some(n) => Ok((i, n)),
None => Err(Err::Error(())),
}
}
}
}
proptest! {
#[test]
#[cfg(feature = "std")]
fn floats(s in "\\PC*") {
println!("testing {}", s);
let res1 = parse_f64(&s);
let res2 = f64::<_, (), false>(s.as_str());
assert_eq!(res1, res2);
}
}
}
mod streaming {
use super::*;
use crate::combinator::opt;
use crate::error::ErrorKind;
use crate::input::ParseTo;
use crate::input::Streaming;
use crate::sequence::pair;
use crate::{Err, IResult, Needed};
use proptest::prelude::*;
macro_rules! assert_parse(
($left: expr, $right: expr) => {
let res: $crate::IResult<_, _, (_, ErrorKind)> = $left;
assert_eq!(res, $right);
};
);
#[test]
fn character() {
let a: &[u8] = b"abcd";
let b: &[u8] = b"1234";
let c: &[u8] = b"a123";
let d: &[u8] = "azé12".as_bytes();
let e: &[u8] = b" ";
let f: &[u8] = b" ;";
assert_parse!(alpha1(Streaming(a)), Err(Err::Incomplete(Needed::new(1))));
assert_eq!(
alpha1(Streaming(b)),
Err(Err::Error((Streaming(b), ErrorKind::Alpha)))
);
assert_eq!(
alpha1::<_, (_, ErrorKind), true>(Streaming(c)),
Ok((Streaming(&c[1..]), &b"a"[..]))
);
assert_eq!(
alpha1::<_, (_, ErrorKind), true>(Streaming(d)),
Ok((Streaming("é12".as_bytes()), &b"az"[..]))
);
assert_eq!(
digit1(Streaming(a)),
Err(Err::Error((Streaming(a), ErrorKind::Digit)))
);
assert_eq!(
digit1::<_, (_, ErrorKind), true>(Streaming(b)),
Err(Err::Incomplete(Needed::new(1)))
);
assert_eq!(
digit1(Streaming(c)),
Err(Err::Error((Streaming(c), ErrorKind::Digit)))
);
assert_eq!(
digit1(Streaming(d)),
Err(Err::Error((Streaming(d), ErrorKind::Digit)))
);
assert_eq!(
hex_digit1::<_, (_, ErrorKind), true>(Streaming(a)),
Err(Err::Incomplete(Needed::new(1)))
);
assert_eq!(
hex_digit1::<_, (_, ErrorKind), true>(Streaming(b)),
Err(Err::Incomplete(Needed::new(1)))
);
assert_eq!(
hex_digit1::<_, (_, ErrorKind), true>(Streaming(c)),
Err(Err::Incomplete(Needed::new(1)))
);
assert_eq!(
hex_digit1::<_, (_, ErrorKind), true>(Streaming(d)),
Ok((Streaming("zé12".as_bytes()), &b"a"[..]))
);
assert_eq!(
hex_digit1(Streaming(e)),
Err(Err::Error((Streaming(e), ErrorKind::HexDigit)))
);
assert_eq!(
oct_digit1(Streaming(a)),
Err(Err::Error((Streaming(a), ErrorKind::OctDigit)))
);
assert_eq!(
oct_digit1::<_, (_, ErrorKind), true>(Streaming(b)),
Err(Err::Incomplete(Needed::new(1)))
);
assert_eq!(
oct_digit1(Streaming(c)),
Err(Err::Error((Streaming(c), ErrorKind::OctDigit)))
);
assert_eq!(
oct_digit1(Streaming(d)),
Err(Err::Error((Streaming(d), ErrorKind::OctDigit)))
);
assert_eq!(
alphanumeric1::<_, (_, ErrorKind), true>(Streaming(a)),
Err(Err::Incomplete(Needed::new(1)))
);
assert_eq!(
alphanumeric1::<_, (_, ErrorKind), true>(Streaming(c)),
Err(Err::Incomplete(Needed::new(1)))
);
assert_eq!(
alphanumeric1::<_, (_, ErrorKind), true>(Streaming(d)),
Ok((Streaming("é12".as_bytes()), &b"az"[..]))
);
assert_eq!(
space1::<_, (_, ErrorKind), true>(Streaming(e)),
Err(Err::Incomplete(Needed::new(1)))
);
assert_eq!(
space1::<_, (_, ErrorKind), true>(Streaming(f)),
Ok((Streaming(&b";"[..]), &b" "[..]))
);
}
#[cfg(feature = "alloc")]
#[test]
fn character_s() {
let a = "abcd";
let b = "1234";
let c = "a123";
let d = "azé12";
let e = " ";
assert_eq!(
alpha1::<_, (_, ErrorKind), true>(Streaming(a)),
Err(Err::Incomplete(Needed::new(1)))
);
assert_eq!(
alpha1(Streaming(b)),
Err(Err::Error((Streaming(b), ErrorKind::Alpha)))
);
assert_eq!(
alpha1::<_, (_, ErrorKind), true>(Streaming(c)),
Ok((Streaming(&c[1..]), &"a"[..]))
);
assert_eq!(
alpha1::<_, (_, ErrorKind), true>(Streaming(d)),
Ok((Streaming("é12"), &"az"[..]))
);
assert_eq!(
digit1(Streaming(a)),
Err(Err::Error((Streaming(a), ErrorKind::Digit)))
);
assert_eq!(
digit1::<_, (_, ErrorKind), true>(Streaming(b)),
Err(Err::Incomplete(Needed::new(1)))
);
assert_eq!(
digit1(Streaming(c)),
Err(Err::Error((Streaming(c), ErrorKind::Digit)))
);
assert_eq!(
digit1(Streaming(d)),
Err(Err::Error((Streaming(d), ErrorKind::Digit)))
);
assert_eq!(
hex_digit1::<_, (_, ErrorKind), true>(Streaming(a)),
Err(Err::Incomplete(Needed::new(1)))
);
assert_eq!(
hex_digit1::<_, (_, ErrorKind), true>(Streaming(b)),
Err(Err::Incomplete(Needed::new(1)))
);
assert_eq!(
hex_digit1::<_, (_, ErrorKind), true>(Streaming(c)),
Err(Err::Incomplete(Needed::new(1)))
);
assert_eq!(
hex_digit1::<_, (_, ErrorKind), true>(Streaming(d)),
Ok((Streaming("zé12"), &"a"[..]))
);
assert_eq!(
hex_digit1(Streaming(e)),
Err(Err::Error((Streaming(e), ErrorKind::HexDigit)))
);
assert_eq!(
oct_digit1(Streaming(a)),
Err(Err::Error((Streaming(a), ErrorKind::OctDigit)))
);
assert_eq!(
oct_digit1::<_, (_, ErrorKind), true>(Streaming(b)),
Err(Err::Incomplete(Needed::new(1)))
);
assert_eq!(
oct_digit1(Streaming(c)),
Err(Err::Error((Streaming(c), ErrorKind::OctDigit)))
);
assert_eq!(
oct_digit1(Streaming(d)),
Err(Err::Error((Streaming(d), ErrorKind::OctDigit)))
);
assert_eq!(
alphanumeric1::<_, (_, ErrorKind), true>(Streaming(a)),
Err(Err::Incomplete(Needed::new(1)))
);
assert_eq!(
alphanumeric1::<_, (_, ErrorKind), true>(Streaming(c)),
Err(Err::Incomplete(Needed::new(1)))
);
assert_eq!(
alphanumeric1::<_, (_, ErrorKind), true>(Streaming(d)),
Ok((Streaming("é12"), "az"))
);
assert_eq!(
space1::<_, (_, ErrorKind), true>(Streaming(e)),
Err(Err::Incomplete(Needed::new(1)))
);
}
use crate::input::Offset;
#[test]
fn offset() {
let a = &b"abcd;"[..];
let b = &b"1234;"[..];
let c = &b"a123;"[..];
let d = &b" \t;"[..];
let e = &b" \t\r\n;"[..];
let f = &b"123abcDEF;"[..];
match alpha1::<_, (_, ErrorKind), true>(Streaming(a)) {
Ok((Streaming(i), _)) => {
assert_eq!(a.offset(i) + i.len(), a.len());
}
_ => panic!("wrong return type in offset test for alpha"),
}
match digit1::<_, (_, ErrorKind), true>(Streaming(b)) {
Ok((Streaming(i), _)) => {
assert_eq!(b.offset(i) + i.len(), b.len());
}
_ => panic!("wrong return type in offset test for digit"),
}
match alphanumeric1::<_, (_, ErrorKind), true>(Streaming(c)) {
Ok((Streaming(i), _)) => {
assert_eq!(c.offset(i) + i.len(), c.len());
}
_ => panic!("wrong return type in offset test for alphanumeric"),
}
match space1::<_, (_, ErrorKind), true>(Streaming(d)) {
Ok((Streaming(i), _)) => {
assert_eq!(d.offset(i) + i.len(), d.len());
}
_ => panic!("wrong return type in offset test for space"),
}
match multispace1::<_, (_, ErrorKind), true>(Streaming(e)) {
Ok((Streaming(i), _)) => {
assert_eq!(e.offset(i) + i.len(), e.len());
}
_ => panic!("wrong return type in offset test for multispace"),
}
match hex_digit1::<_, (_, ErrorKind), true>(Streaming(f)) {
Ok((Streaming(i), _)) => {
assert_eq!(f.offset(i) + i.len(), f.len());
}
_ => panic!("wrong return type in offset test for hex_digit"),
}
match oct_digit1::<_, (_, ErrorKind), true>(Streaming(f)) {
Ok((Streaming(i), _)) => {
assert_eq!(f.offset(i) + i.len(), f.len());
}
_ => panic!("wrong return type in offset test for oct_digit"),
}
}
#[test]
fn is_not_line_ending_bytes() {
let a: &[u8] = b"ab12cd\nefgh";
assert_eq!(
not_line_ending::<_, (_, ErrorKind), true>(Streaming(a)),
Ok((Streaming(&b"\nefgh"[..]), &b"ab12cd"[..]))
);
let b: &[u8] = b"ab12cd\nefgh\nijkl";
assert_eq!(
not_line_ending::<_, (_, ErrorKind), true>(Streaming(b)),
Ok((Streaming(&b"\nefgh\nijkl"[..]), &b"ab12cd"[..]))
);
let c: &[u8] = b"ab12cd\r\nefgh\nijkl";
assert_eq!(
not_line_ending::<_, (_, ErrorKind), true>(Streaming(c)),
Ok((Streaming(&b"\r\nefgh\nijkl"[..]), &b"ab12cd"[..]))
);
let d: &[u8] = b"ab12cd";
assert_eq!(
not_line_ending::<_, (_, ErrorKind), true>(Streaming(d)),
Err(Err::Incomplete(Needed::Unknown))
);
}
#[test]
fn is_not_line_ending_str() {
let f = "βèƒôřè\rÂßÇáƒĆèř";
assert_eq!(
not_line_ending(Streaming(f)),
Err(Err::Error((Streaming(f), ErrorKind::Tag)))
);
let g2: &str = "ab12cd";
assert_eq!(
not_line_ending::<_, (_, ErrorKind), true>(Streaming(g2)),
Err(Err::Incomplete(Needed::Unknown))
);
}
#[test]
fn hex_digit_test() {
let i = &b"0123456789abcdefABCDEF;"[..];
assert_parse!(
hex_digit1(Streaming(i)),
Ok((Streaming(&b";"[..]), &i[..i.len() - 1]))
);
let i = &b"g"[..];
assert_parse!(
hex_digit1(Streaming(i)),
Err(Err::Error(error_position!(
Streaming(i),
ErrorKind::HexDigit
)))
);
let i = &b"G"[..];
assert_parse!(
hex_digit1(Streaming(i)),
Err(Err::Error(error_position!(
Streaming(i),
ErrorKind::HexDigit
)))
);
assert!(AsChar::is_hex_digit(b'0'));
assert!(AsChar::is_hex_digit(b'9'));
assert!(AsChar::is_hex_digit(b'a'));
assert!(AsChar::is_hex_digit(b'f'));
assert!(AsChar::is_hex_digit(b'A'));
assert!(AsChar::is_hex_digit(b'F'));
assert!(!AsChar::is_hex_digit(b'g'));
assert!(!AsChar::is_hex_digit(b'G'));
assert!(!AsChar::is_hex_digit(b'/'));
assert!(!AsChar::is_hex_digit(b':'));
assert!(!AsChar::is_hex_digit(b'@'));
assert!(!AsChar::is_hex_digit(b'\x60'));
}
#[test]
fn oct_digit_test() {
let i = &b"01234567;"[..];
assert_parse!(
oct_digit1(Streaming(i)),
Ok((Streaming(&b";"[..]), &i[..i.len() - 1]))
);
let i = &b"8"[..];
assert_parse!(
oct_digit1(Streaming(i)),
Err(Err::Error(error_position!(
Streaming(i),
ErrorKind::OctDigit
)))
);
assert!(AsChar::is_oct_digit(b'0'));
assert!(AsChar::is_oct_digit(b'7'));
assert!(!AsChar::is_oct_digit(b'8'));
assert!(!AsChar::is_oct_digit(b'9'));
assert!(!AsChar::is_oct_digit(b'a'));
assert!(!AsChar::is_oct_digit(b'A'));
assert!(!AsChar::is_oct_digit(b'/'));
assert!(!AsChar::is_oct_digit(b':'));
assert!(!AsChar::is_oct_digit(b'@'));
assert!(!AsChar::is_oct_digit(b'\x60'));
}
#[test]
fn full_line_windows() {
fn take_full_line(i: Streaming<&[u8]>) -> IResult<Streaming<&[u8]>, (&[u8], &[u8])> {
pair(not_line_ending, line_ending)(i)
}
let input = b"abc\r\n";
let output = take_full_line(Streaming(input));
assert_eq!(
output,
Ok((Streaming(&b""[..]), (&b"abc"[..], &b"\r\n"[..])))
);
}
#[test]
fn full_line_unix() {
fn take_full_line(i: Streaming<&[u8]>) -> IResult<Streaming<&[u8]>, (&[u8], &[u8])> {
pair(not_line_ending, line_ending)(i)
}
let input = b"abc\n";
let output = take_full_line(Streaming(input));
assert_eq!(output, Ok((Streaming(&b""[..]), (&b"abc"[..], &b"\n"[..]))));
}
#[test]
fn check_windows_lineending() {
let input = b"\r\n";
let output = line_ending(Streaming(&input[..]));
assert_parse!(output, Ok((Streaming(&b""[..]), &b"\r\n"[..])));
}
#[test]
fn check_unix_lineending() {
let input = b"\n";
let output = line_ending(Streaming(&input[..]));
assert_parse!(output, Ok((Streaming(&b""[..]), &b"\n"[..])));
}
#[test]
fn cr_lf() {
assert_parse!(
crlf(Streaming(&b"\r\na"[..])),
Ok((Streaming(&b"a"[..]), &b"\r\n"[..]))
);
assert_parse!(
crlf(Streaming(&b"\r"[..])),
Err(Err::Incomplete(Needed::new(2)))
);
assert_parse!(
crlf(Streaming(&b"\ra"[..])),
Err(Err::Error(error_position!(
Streaming(&b"\ra"[..]),
ErrorKind::CrLf
)))
);
assert_parse!(crlf(Streaming("\r\na")), Ok((Streaming("a"), "\r\n")));
assert_parse!(crlf(Streaming("\r")), Err(Err::Incomplete(Needed::new(2))));
assert_parse!(
crlf(Streaming("\ra")),
Err(Err::Error(error_position!(
Streaming("\ra"),
ErrorKind::CrLf
)))
);
}
#[test]
fn end_of_line() {
assert_parse!(
line_ending(Streaming(&b"\na"[..])),
Ok((Streaming(&b"a"[..]), &b"\n"[..]))
);
assert_parse!(
line_ending(Streaming(&b"\r\na"[..])),
Ok((Streaming(&b"a"[..]), &b"\r\n"[..]))
);
assert_parse!(
line_ending(Streaming(&b"\r"[..])),
Err(Err::Incomplete(Needed::new(2)))
);
assert_parse!(
line_ending(Streaming(&b"\ra"[..])),
Err(Err::Error(error_position!(
Streaming(&b"\ra"[..]),
ErrorKind::CrLf
)))
);
assert_parse!(line_ending(Streaming("\na")), Ok((Streaming("a"), "\n")));
assert_parse!(
line_ending(Streaming("\r\na")),
Ok((Streaming("a"), "\r\n"))
);
assert_parse!(
line_ending(Streaming("\r")),
Err(Err::Incomplete(Needed::new(2)))
);
assert_parse!(
line_ending(Streaming("\ra")),
Err(Err::Error(error_position!(
Streaming("\ra"),
ErrorKind::CrLf
)))
);
}
fn digit_to_i16(input: Streaming<&str>) -> IResult<Streaming<&str>, i16> {
use crate::bytes::one_of;
let i = input;
let (i, opt_sign) = opt(one_of("+-"))(i)?;
let sign = match opt_sign {
Some('+') => true,
Some('-') => false,
_ => true,
};
let (i, s) = match digit1::<_, crate::error::Error<_>, true>(i) {
Ok((i, s)) => (i, s),
Err(Err::Incomplete(i)) => return Err(Err::Incomplete(i)),
Err(_) => {
return Err(Err::Error(crate::error::Error::from_error_kind(
input,
ErrorKind::Digit,
)))
}
};
match s.parse_to() {
Some(n) => {
if sign {
Ok((i, n))
} else {
Ok((i, -n))
}
}
None => Err(Err::Error(crate::error::Error::from_error_kind(
i,
ErrorKind::Digit,
))),
}
}
fn digit_to_u32(i: Streaming<&str>) -> IResult<Streaming<&str>, u32> {
let (i, s) = digit1(i)?;
match s.parse_to() {
Some(n) => Ok((i, n)),
None => Err(Err::Error(crate::error::Error::from_error_kind(
i,
ErrorKind::Digit,
))),
}
}
proptest! {
#[test]
fn ints(s in "\\PC*") {
let res1 = digit_to_i16(Streaming(&s));
let res2 = i16(Streaming(s.as_str()));
assert_eq!(res1, res2);
}
#[test]
fn uints(s in "\\PC*") {
let res1 = digit_to_u32(Streaming(&s));
let res2 = u32(Streaming(s.as_str()));
assert_eq!(res1, res2);
}
}
}