| use crate::branch::{alt, permutation}; |
| use crate::bytes::tag; |
| use crate::error::ErrorKind; |
| use crate::input::Streaming; |
| #[cfg(feature = "alloc")] |
| use crate::{ |
| error::ParseError, |
| lib::std::{ |
| fmt::Debug, |
| string::{String, ToString}, |
| }, |
| }; |
| use crate::{Err, IResult, Needed}; |
| |
| #[cfg(feature = "alloc")] |
| #[derive(Debug, Clone, PartialEq)] |
| pub struct ErrorStr(String); |
| |
| #[cfg(feature = "alloc")] |
| impl From<u32> for ErrorStr { |
| fn from(i: u32) -> Self { |
| ErrorStr(format!("custom error code: {}", i)) |
| } |
| } |
| |
| #[cfg(feature = "alloc")] |
| impl<'a> From<&'a str> for ErrorStr { |
| fn from(i: &'a str) -> Self { |
| ErrorStr(format!("custom error message: {}", i)) |
| } |
| } |
| |
| #[cfg(feature = "alloc")] |
| impl<I: Debug> ParseError<I> for ErrorStr { |
| fn from_error_kind(input: I, kind: ErrorKind) -> Self { |
| ErrorStr(format!("custom error message: ({:?}, {:?})", input, kind)) |
| } |
| |
| fn append(input: I, kind: ErrorKind, other: Self) -> Self { |
| ErrorStr(format!( |
| "custom error message: ({:?}, {:?}) - {:?}", |
| input, kind, other |
| )) |
| } |
| } |
| |
| #[cfg(feature = "alloc")] |
| #[test] |
| fn alt_test() { |
| fn work(input: &[u8]) -> IResult<&[u8], &[u8], ErrorStr> { |
| Ok((&b""[..], input)) |
| } |
| |
| #[allow(unused_variables)] |
| fn dont_work(input: &[u8]) -> IResult<&[u8], &[u8], ErrorStr> { |
| Err(Err::Error(ErrorStr("abcd".to_string()))) |
| } |
| |
| fn work2(input: &[u8]) -> IResult<&[u8], &[u8], ErrorStr> { |
| Ok((input, &b""[..])) |
| } |
| |
| fn alt1(i: &[u8]) -> IResult<&[u8], &[u8], ErrorStr> { |
| alt((dont_work, dont_work))(i) |
| } |
| fn alt2(i: &[u8]) -> IResult<&[u8], &[u8], ErrorStr> { |
| alt((dont_work, work))(i) |
| } |
| fn alt3(i: &[u8]) -> IResult<&[u8], &[u8], ErrorStr> { |
| alt((dont_work, dont_work, work2, dont_work))(i) |
| } |
| //named!(alt1, alt!(dont_work | dont_work)); |
| //named!(alt2, alt!(dont_work | work)); |
| //named!(alt3, alt!(dont_work | dont_work | work2 | dont_work)); |
| |
| let a = &b"abcd"[..]; |
| assert_eq!( |
| alt1(a), |
| Err(Err::Error(error_node_position!( |
| a, |
| ErrorKind::Alt, |
| ErrorStr("abcd".to_string()) |
| ))) |
| ); |
| assert_eq!(alt2(a), Ok((&b""[..], a))); |
| assert_eq!(alt3(a), Ok((a, &b""[..]))); |
| |
| fn alt4(i: &[u8]) -> IResult<&[u8], &[u8]> { |
| alt((tag("abcd"), tag("efgh")))(i) |
| } |
| let b = &b"efgh"[..]; |
| assert_eq!(alt4(a), Ok((&b""[..], a))); |
| assert_eq!(alt4(b), Ok((&b""[..], b))); |
| } |
| |
| #[test] |
| fn alt_incomplete() { |
| fn alt1(i: Streaming<&[u8]>) -> IResult<Streaming<&[u8]>, &[u8]> { |
| alt((tag("a"), tag("bc"), tag("def")))(i) |
| } |
| |
| let a = &b""[..]; |
| assert_eq!(alt1(Streaming(a)), Err(Err::Incomplete(Needed::new(1)))); |
| let a = &b"b"[..]; |
| assert_eq!(alt1(Streaming(a)), Err(Err::Incomplete(Needed::new(1)))); |
| let a = &b"bcd"[..]; |
| assert_eq!(alt1(Streaming(a)), Ok((Streaming(&b"d"[..]), &b"bc"[..]))); |
| let a = &b"cde"[..]; |
| assert_eq!( |
| alt1(Streaming(a)), |
| Err(Err::Error(error_position!(Streaming(a), ErrorKind::Tag))) |
| ); |
| let a = &b"de"[..]; |
| assert_eq!(alt1(Streaming(a)), Err(Err::Incomplete(Needed::new(1)))); |
| let a = &b"defg"[..]; |
| assert_eq!(alt1(Streaming(a)), Ok((Streaming(&b"g"[..]), &b"def"[..]))); |
| } |
| |
| #[test] |
| fn permutation_test() { |
| fn perm(i: Streaming<&[u8]>) -> IResult<Streaming<&[u8]>, (&[u8], &[u8], &[u8])> { |
| permutation((tag("abcd"), tag("efg"), tag("hi")))(i) |
| } |
| |
| let expected = (&b"abcd"[..], &b"efg"[..], &b"hi"[..]); |
| |
| let a = &b"abcdefghijk"[..]; |
| assert_eq!(perm(Streaming(a)), Ok((Streaming(&b"jk"[..]), expected))); |
| let b = &b"efgabcdhijk"[..]; |
| assert_eq!(perm(Streaming(b)), Ok((Streaming(&b"jk"[..]), expected))); |
| let c = &b"hiefgabcdjk"[..]; |
| assert_eq!(perm(Streaming(c)), Ok((Streaming(&b"jk"[..]), expected))); |
| |
| let d = &b"efgxyzabcdefghi"[..]; |
| assert_eq!( |
| perm(Streaming(d)), |
| Err(Err::Error(error_node_position!( |
| Streaming(&b"efgxyzabcdefghi"[..]), |
| ErrorKind::Permutation, |
| error_position!(Streaming(&b"xyzabcdefghi"[..]), ErrorKind::Tag) |
| ))) |
| ); |
| |
| let e = &b"efgabc"[..]; |
| assert_eq!(perm(Streaming(e)), Err(Err::Incomplete(Needed::new(1)))); |
| } |