#![allow(deprecated)]
use crate::error::ErrMode;
use crate::error::ErrorKind;
use crate::error::ParseError;
use crate::lib::std::result::Result::Ok;
use crate::stream::{
split_at_offset1_complete, split_at_offset_complete, Compare, CompareResult, ContainsToken,
FindSlice, Offset, SliceLen, Stream, ToUsize,
};
use crate::{IResult, Parser};
pub(crate) fn any<I, E: ParseError<I>>(input: I) -> IResult<I, <I as Stream>::Token, E>
where
I: Stream,
{
input
.next_token()
.ok_or_else(|| ErrMode::from_error_kind(input, ErrorKind::Eof))
}
#[deprecated(since = "0.1.0", note = "Replaced with `winnow::bytes::tag`")]
pub fn tag<T, I, Error: ParseError<I>>(
tag: T,
) -> impl Fn(I) -> IResult<I, <I as Stream>::Slice, Error>
where
I: Stream + Compare<T>,
T: SliceLen + Clone,
{
move |i: I| tag_internal(i, tag.clone())
}
pub(crate) fn tag_internal<T, I, Error: ParseError<I>>(
i: I,
t: T,
) -> IResult<I, <I as Stream>::Slice, Error>
where
I: Stream + Compare<T>,
T: SliceLen,
{
let tag_len = t.slice_len();
match i.compare(t) {
CompareResult::Ok => Ok(i.next_slice(tag_len)),
CompareResult::Incomplete | CompareResult::Error => {
let e: ErrorKind = ErrorKind::Tag;
Err(ErrMode::from_error_kind(i, e))
}
}
}
#[deprecated(since = "0.1.0", note = "Replaced with `winnow::bytes::tag_no_case`")]
pub fn tag_no_case<T, I, Error: ParseError<I>>(
tag: T,
) -> impl Fn(I) -> IResult<I, <I as Stream>::Slice, Error>
where
I: Stream + Compare<T>,
T: SliceLen + Clone,
{
move |i: I| tag_no_case_internal(i, tag.clone())
}
pub(crate) fn tag_no_case_internal<T, I, Error: ParseError<I>>(
i: I,
t: T,
) -> IResult<I, <I as Stream>::Slice, Error>
where
I: Stream + Compare<T>,
T: SliceLen,
{
let tag_len = t.slice_len();
match (i).compare_no_case(t) {
CompareResult::Ok => Ok(i.next_slice(tag_len)),
CompareResult::Incomplete | CompareResult::Error => {
let e: ErrorKind = ErrorKind::Tag;
Err(ErrMode::from_error_kind(i, e))
}
}
}
pub(crate) fn one_of_internal<I, T, E: ParseError<I>>(
input: I,
list: &T,
) -> IResult<I, <I as Stream>::Token, E>
where
I: Stream,
<I as Stream>::Token: Copy,
T: ContainsToken<<I as Stream>::Token>,
{
input
.next_token()
.filter(|(_, t)| list.contains_token(*t))
.ok_or_else(|| ErrMode::from_error_kind(input, ErrorKind::OneOf))
}
pub(crate) fn none_of_internal<I, T, E: ParseError<I>>(
input: I,
list: &T,
) -> IResult<I, <I as Stream>::Token, E>
where
I: Stream,
<I as Stream>::Token: Copy,
T: ContainsToken<<I as Stream>::Token>,
{
input
.next_token()
.filter(|(_, t)| !list.contains_token(*t))
.ok_or_else(|| ErrMode::from_error_kind(input, ErrorKind::NoneOf))
}
#[deprecated(since = "0.1.0", note = "Replaced with `winnow::bytes::take_till1`")]
pub fn is_not<T, I, Error: ParseError<I>>(
arr: T,
) -> impl Fn(I) -> IResult<I, <I as Stream>::Slice, Error>
where
I: Stream,
T: ContainsToken<<I as Stream>::Token>,
{
move |i: I| is_not_internal(i, &arr)
}
pub(crate) fn is_not_internal<T, I, Error: ParseError<I>>(
i: I,
arr: &T,
) -> IResult<I, <I as Stream>::Slice, Error>
where
I: Stream,
T: ContainsToken<<I as Stream>::Token>,
{
let e: ErrorKind = ErrorKind::IsNot;
split_at_offset1_complete(&i, |c| arr.contains_token(c), e)
}
#[deprecated(since = "0.1.0", note = "Replaced with `winnow::bytes::take_while1`")]
pub fn is_a<T, I, Error: ParseError<I>>(
arr: T,
) -> impl Fn(I) -> IResult<I, <I as Stream>::Slice, Error>
where
I: Stream,
T: ContainsToken<<I as Stream>::Token>,
{
move |i: I| is_a_internal(i, &arr)
}
pub(crate) fn is_a_internal<T, I, Error: ParseError<I>>(
i: I,
arr: &T,
) -> IResult<I, <I as Stream>::Slice, Error>
where
I: Stream,
T: ContainsToken<<I as Stream>::Token>,
{
let e: ErrorKind = ErrorKind::IsA;
split_at_offset1_complete(&i, |c| !arr.contains_token(c), e)
}
#[deprecated(since = "0.1.0", note = "Replaced with `winnow::bytes::take_while0`")]
pub fn take_while<T, I, Error: ParseError<I>>(
list: T,
) -> impl Fn(I) -> IResult<I, <I as Stream>::Slice, Error>
where
I: Stream,
T: ContainsToken<<I as Stream>::Token>,
{
move |i: I| take_while_internal(i, &list)
}
pub(crate) fn take_while_internal<T, I, Error: ParseError<I>>(
i: I,
list: &T,
) -> IResult<I, <I as Stream>::Slice, Error>
where
I: Stream,
T: ContainsToken<<I as Stream>::Token>,
{
split_at_offset_complete(&i, |c| !list.contains_token(c))
}
#[deprecated(since = "0.1.0", note = "Replaced with `winnow::bytes::take_while1`")]
pub fn take_while1<T, I, Error: ParseError<I>>(
list: T,
) -> impl Fn(I) -> IResult<I, <I as Stream>::Slice, Error>
where
I: Stream,
T: ContainsToken<<I as Stream>::Token>,
{
move |i: I| take_while1_internal(i, &list)
}
pub(crate) fn take_while1_internal<T, I, Error: ParseError<I>>(
i: I,
list: &T,
) -> IResult<I, <I as Stream>::Slice, Error>
where
I: Stream,
T: ContainsToken<<I as Stream>::Token>,
{
let e: ErrorKind = ErrorKind::TakeWhile1;
split_at_offset1_complete(&i, |c| !list.contains_token(c), e)
}
#[deprecated(
since = "0.1.0",
note = "Replaced with `winnow::bytes::take_while_m_n`"
)]
pub fn take_while_m_n<T, I, Error: ParseError<I>>(
m: usize,
n: usize,
list: T,
) -> impl Fn(I) -> IResult<I, <I as Stream>::Slice, Error>
where
I: Stream,
T: ContainsToken<<I as Stream>::Token>,
{
move |i: I| take_while_m_n_internal(i, m, n, &list)
}
pub(crate) fn take_while_m_n_internal<T, I, Error: ParseError<I>>(
input: I,
m: usize,
n: usize,
list: &T,
) -> IResult<I, <I as Stream>::Slice, Error>
where
I: Stream,
T: ContainsToken<<I as Stream>::Token>,
{
if n < m {
return Err(ErrMode::from_error_kind(input, ErrorKind::TakeWhileMN));
}
let mut final_count = 0;
for (processed, (offset, token)) in input.iter_offsets().enumerate() {
if !list.contains_token(token) {
if processed < m {
return Err(ErrMode::from_error_kind(input, ErrorKind::TakeWhileMN));
} else {
return Ok(input.next_slice(offset));
}
} else {
if processed == n {
return Ok(input.next_slice(offset));
}
final_count = processed + 1;
}
}
if m <= final_count {
Ok(input.next_slice(input.eof_offset()))
} else {
Err(ErrMode::from_error_kind(input, ErrorKind::TakeWhileMN))
}
}
#[deprecated(since = "0.1.0", note = "Replaced with `winnow::bytes::take_till0`")]
#[allow(clippy::redundant_closure)]
pub fn take_till<T, I, Error: ParseError<I>>(
list: T,
) -> impl Fn(I) -> IResult<I, <I as Stream>::Slice, Error>
where
I: Stream,
T: ContainsToken<<I as Stream>::Token>,
{
move |i: I| take_till_internal(i, &list)
}
pub(crate) fn take_till_internal<T, I, Error: ParseError<I>>(
i: I,
list: &T,
) -> IResult<I, <I as Stream>::Slice, Error>
where
I: Stream,
T: ContainsToken<<I as Stream>::Token>,
{
split_at_offset_complete(&i, |c| list.contains_token(c))
}
#[deprecated(since = "0.1.0", note = "Replaced with `winnow::bytes::take_till1`")]
#[allow(clippy::redundant_closure)]
pub fn take_till1<T, I, Error: ParseError<I>>(
list: T,
) -> impl Fn(I) -> IResult<I, <I as Stream>::Slice, Error>
where
I: Stream,
T: ContainsToken<<I as Stream>::Token>,
{
move |i: I| take_till1_internal(i, &list)
}
pub(crate) fn take_till1_internal<T, I, Error: ParseError<I>>(
i: I,
list: &T,
) -> IResult<I, <I as Stream>::Slice, Error>
where
I: Stream,
T: ContainsToken<<I as Stream>::Token>,
{
let e: ErrorKind = ErrorKind::TakeTill1;
split_at_offset1_complete(&i, |c| list.contains_token(c), e)
}
#[deprecated(since = "0.1.0", note = "Replaced with `winnow::bytes::take`")]
pub fn take<C, I, Error: ParseError<I>>(
count: C,
) -> impl Fn(I) -> IResult<I, <I as Stream>::Slice, Error>
where
I: Stream,
C: ToUsize,
{
let c = count.to_usize();
move |i: I| take_internal(i, c)
}
pub(crate) fn take_internal<I, Error: ParseError<I>>(
i: I,
c: usize,
) -> IResult<I, <I as Stream>::Slice, Error>
where
I: Stream,
{
match i.offset_at(c) {
Ok(offset) => Ok(i.next_slice(offset)),
Err(_needed) => Err(ErrMode::from_error_kind(i, ErrorKind::Eof)),
}
}
#[deprecated(since = "0.1.0", note = "Replaced with `winnow::bytes::take_until0`")]
pub fn take_until<T, I, Error: ParseError<I>>(
tag: T,
) -> impl Fn(I) -> IResult<I, <I as Stream>::Slice, Error>
where
I: Stream + FindSlice<T>,
T: SliceLen + Clone,
{
move |i: I| take_until_internal(i, tag.clone())
}
pub(crate) fn take_until_internal<T, I, Error: ParseError<I>>(
i: I,
t: T,
) -> IResult<I, <I as Stream>::Slice, Error>
where
I: Stream + FindSlice<T>,
T: SliceLen,
{
match i.find_slice(t) {
Some(offset) => Ok(i.next_slice(offset)),
None => Err(ErrMode::from_error_kind(i, ErrorKind::TakeUntil)),
}
}
#[deprecated(since = "0.1.0", note = "Replaced with `winnow::bytes::take_until1`")]
pub fn take_until1<T, I, Error: ParseError<I>>(
tag: T,
) -> impl Fn(I) -> IResult<I, <I as Stream>::Slice, Error>
where
I: Stream + FindSlice<T>,
T: SliceLen + Clone,
{
move |i: I| take_until1_internal(i, tag.clone())
}
pub(crate) fn take_until1_internal<T, I, Error: ParseError<I>>(
i: I,
t: T,
) -> IResult<I, <I as Stream>::Slice, Error>
where
I: Stream + FindSlice<T>,
T: SliceLen,
{
match i.find_slice(t) {
None | Some(0) => Err(ErrMode::from_error_kind(i, ErrorKind::TakeUntil)),
Some(offset) => Ok(i.next_slice(offset)),
}
}
#[deprecated(since = "0.1.0", note = "Replaced with `winnow::character::escaped`")]
pub fn escaped<'a, I: 'a, Error, F, G, O1, O2>(
mut normal: F,
control_char: char,
mut escapable: G,
) -> impl FnMut(I) -> IResult<I, <I as Stream>::Slice, Error>
where
I: Stream + Offset,
<I as Stream>::Token: crate::stream::AsChar,
F: Parser<I, O1, Error>,
G: Parser<I, O2, Error>,
Error: ParseError<I>,
{
move |input: I| escaped_internal(input, &mut normal, control_char, &mut escapable)
}
pub(crate) fn escaped_internal<'a, I: 'a, Error, F, G, O1, O2>(
input: I,
normal: &mut F,
control_char: char,
escapable: &mut G,
) -> IResult<I, <I as Stream>::Slice, Error>
where
I: Stream + Offset,
<I as Stream>::Token: crate::stream::AsChar,
F: Parser<I, O1, Error>,
G: Parser<I, O2, Error>,
Error: ParseError<I>,
{
use crate::stream::AsChar;
let mut i = input.clone();
while i.eof_offset() > 0 {
let current_len = i.eof_offset();
match normal.parse_next(i.clone()) {
Ok((i2, _)) => {
if i2.eof_offset() == 0 {
return Ok(input.next_slice(input.eof_offset()));
} else if i2.eof_offset() == current_len {
let offset = input.offset_to(&i2);
return Ok(input.next_slice(offset));
} else {
i = i2;
}
}
Err(ErrMode::Backtrack(_)) => {
if i.next_token().expect("eof_offset > 0").1.as_char() == control_char {
let next = control_char.len_utf8();
if next >= i.eof_offset() {
return Err(ErrMode::from_error_kind(input, ErrorKind::Escaped));
} else {
match escapable.parse_next(i.next_slice(next).0) {
Ok((i2, _)) => {
if i2.eof_offset() == 0 {
return Ok(input.next_slice(input.eof_offset()));
} else {
i = i2;
}
}
Err(e) => return Err(e),
}
}
} else {
let offset = input.offset_to(&i);
if offset == 0 {
return Err(ErrMode::from_error_kind(input, ErrorKind::Escaped));
}
return Ok(input.next_slice(offset));
}
}
Err(e) => {
return Err(e);
}
}
}
Ok(input.next_slice(input.eof_offset()))
}
#[cfg(feature = "alloc")]
#[deprecated(
since = "0.1.0",
note = "Replaced with `winnow::character::escaped_transform`"
)]
pub fn escaped_transform<I, Error, F, G, Output>(
mut normal: F,
control_char: char,
mut transform: G,
) -> impl FnMut(I) -> IResult<I, Output, Error>
where
I: Stream + Offset,
<I as Stream>::Token: crate::stream::AsChar,
Output: crate::stream::Accumulate<<I as Stream>::Slice>,
F: Parser<I, <I as Stream>::Slice, Error>,
G: Parser<I, <I as Stream>::Slice, Error>,
Error: ParseError<I>,
{
move |input: I| escaped_transform_internal(input, &mut normal, control_char, &mut transform)
}
#[cfg(feature = "alloc")]
pub(crate) fn escaped_transform_internal<I, Error, F, G, Output>(
input: I,
normal: &mut F,
control_char: char,
transform: &mut G,
) -> IResult<I, Output, Error>
where
I: Stream + Offset,
<I as Stream>::Token: crate::stream::AsChar,
Output: crate::stream::Accumulate<<I as Stream>::Slice>,
F: Parser<I, <I as Stream>::Slice, Error>,
G: Parser<I, <I as Stream>::Slice, Error>,
Error: ParseError<I>,
{
use crate::stream::AsChar;
let mut offset = 0;
let mut res = Output::initial(Some(input.eof_offset()));
let i = input.clone();
while offset < i.eof_offset() {
let current_len = i.eof_offset();
let (remainder, _) = i.next_slice(offset);
match normal.parse_next(remainder.clone()) {
Ok((i2, o)) => {
res.accumulate(o);
if i2.eof_offset() == 0 {
return Ok((i.next_slice(i.eof_offset()).0, res));
} else if i2.eof_offset() == current_len {
return Ok((remainder, res));
} else {
offset = input.offset_to(&i2);
}
}
Err(ErrMode::Backtrack(_)) => {
if remainder.next_token().expect("eof_offset > 0").1.as_char() == control_char {
let next = offset + control_char.len_utf8();
let eof_offset = input.eof_offset();
if next >= eof_offset {
return Err(ErrMode::from_error_kind(
remainder,
ErrorKind::EscapedTransform,
));
} else {
match transform.parse_next(i.next_slice(next).0) {
Ok((i2, o)) => {
res.accumulate(o);
if i2.eof_offset() == 0 {
return Ok((i.next_slice(i.eof_offset()).0, res));
} else {
offset = input.offset_to(&i2);
}
}
Err(e) => return Err(e),
}
}
} else {
if offset == 0 {
return Err(ErrMode::from_error_kind(
remainder,
ErrorKind::EscapedTransform,
));
}
return Ok((remainder, res));
}
}
Err(e) => return Err(e),
}
}
Ok((input.next_slice(offset).0, res))
}
#[cfg(test)]
mod tests {
use super::*;
#[cfg(feature = "alloc")]
use crate::{
branch::alt,
combinator::{map, value},
lib::std::string::String,
lib::std::vec::Vec,
};
#[test]
fn complete_take_while_m_n_utf8_all_matching() {
let result: IResult<&str, &str> =
super::take_while_m_n(1, 4, |c: char| c.is_alphabetic())("øn");
assert_eq!(result, Ok(("", "øn")));
}
#[test]
fn complete_take_while_m_n_utf8_all_matching_substring() {
let result: IResult<&str, &str> =
super::take_while_m_n(1, 1, |c: char| c.is_alphabetic())("øn");
assert_eq!(result, Ok(("n", "ø")));
}
fn escaped_string(input: &str) -> IResult<&str, &str> {
use crate::character::complete::{alpha0, one_of};
escaped(alpha0, '\\', one_of("n"))(input)
}
#[test]
fn escaped_hang() {
escaped_string("7").unwrap();
escaped_string("a7").unwrap();
}
fn unquote(input: &str) -> IResult<&str, &str> {
use crate::bytes::complete::*;
use crate::character::complete::*;
use crate::combinator::opt;
use crate::sequence::delimited;
delimited(
char('"'),
escaped(opt(none_of(r#"\""#)), '\\', one_of(r#"\"rnt"#)),
char('"'),
)(input)
}
#[test]
fn escaped_hang_1118() {
assert_eq!(unquote(r#""""#), Ok(("", "")));
}
#[cfg(feature = "alloc")]
#[allow(unused_variables)]
#[test]
fn escaping() {
use crate::character::complete::one_of;
use crate::character::complete::{alpha1 as alpha, digit1 as digit};
fn esc(i: &[u8]) -> IResult<&[u8], &[u8]> {
escaped(alpha, '\\', one_of("\"n\\"))(i)
}
assert_eq!(esc(&b"abcd;"[..]), Ok((&b";"[..], &b"abcd"[..])));
assert_eq!(esc(&b"ab\\\"cd;"[..]), Ok((&b";"[..], &b"ab\\\"cd"[..])));
assert_eq!(esc(&b"\\\"abcd;"[..]), Ok((&b";"[..], &b"\\\"abcd"[..])));
assert_eq!(esc(&b"\\n;"[..]), Ok((&b";"[..], &b"\\n"[..])));
assert_eq!(esc(&b"ab\\\"12"[..]), Ok((&b"12"[..], &b"ab\\\""[..])));
assert_eq!(
esc(&b"AB\\"[..]),
Err(ErrMode::Backtrack(error_position!(
&b"AB\\"[..],
ErrorKind::Escaped
)))
);
assert_eq!(
esc(&b"AB\\A"[..]),
Err(ErrMode::Backtrack(error_node_position!(
&b"AB\\A"[..],
ErrorKind::Escaped,
error_position!(&b"A"[..], ErrorKind::OneOf)
)))
);
fn esc2(i: &[u8]) -> IResult<&[u8], &[u8]> {
escaped(digit, '\\', one_of("\"n\\"))(i)
}
assert_eq!(esc2(&b"12\\nnn34"[..]), Ok((&b"nn34"[..], &b"12\\n"[..])));
}
#[cfg(feature = "alloc")]
#[test]
fn escaping_str() {
use crate::character::complete::one_of;
use crate::character::complete::{alpha1 as alpha, digit1 as digit};
fn esc(i: &str) -> IResult<&str, &str> {
escaped(alpha, '\\', one_of("\"n\\"))(i)
}
assert_eq!(esc("abcd;"), Ok((";", "abcd")));
assert_eq!(esc("ab\\\"cd;"), Ok((";", "ab\\\"cd")));
assert_eq!(esc("\\\"abcd;"), Ok((";", "\\\"abcd")));
assert_eq!(esc("\\n;"), Ok((";", "\\n")));
assert_eq!(esc("ab\\\"12"), Ok(("12", "ab\\\"")));
assert_eq!(
esc("AB\\"),
Err(ErrMode::Backtrack(error_position!(
"AB\\",
ErrorKind::Escaped
)))
);
assert_eq!(
esc("AB\\A"),
Err(ErrMode::Backtrack(error_node_position!(
"AB\\A",
ErrorKind::Escaped,
error_position!("A", ErrorKind::OneOf)
)))
);
fn esc2(i: &str) -> IResult<&str, &str> {
escaped(digit, '\\', one_of("\"n\\"))(i)
}
assert_eq!(esc2("12\\nnn34"), Ok(("nn34", "12\\n")));
fn esc3(i: &str) -> IResult<&str, &str> {
escaped(alpha, '\u{241b}', one_of("\"n"))(i)
}
assert_eq!(esc3("ab␛ncd;"), Ok((";", "ab␛ncd")));
}
#[cfg(feature = "alloc")]
fn to_s(i: Vec<u8>) -> String {
String::from_utf8_lossy(&i).into_owned()
}
#[cfg(feature = "alloc")]
#[test]
fn escape_transform() {
use crate::character::complete::alpha1 as alpha;
fn esc(i: &[u8]) -> IResult<&[u8], String> {
map(
escaped_transform(
alpha,
'\\',
alt((
value(&b"\\"[..], tag("\\")),
value(&b"\""[..], tag("\"")),
value(&b"\n"[..], tag("n")),
)),
),
to_s,
)(i)
}
assert_eq!(esc(&b"abcd;"[..]), Ok((&b";"[..], String::from("abcd"))));
assert_eq!(
esc(&b"ab\\\"cd;"[..]),
Ok((&b";"[..], String::from("ab\"cd")))
);
assert_eq!(
esc(&b"\\\"abcd;"[..]),
Ok((&b";"[..], String::from("\"abcd")))
);
assert_eq!(esc(&b"\\n;"[..]), Ok((&b";"[..], String::from("\n"))));
assert_eq!(
esc(&b"ab\\\"12"[..]),
Ok((&b"12"[..], String::from("ab\"")))
);
assert_eq!(
esc(&b"AB\\"[..]),
Err(ErrMode::Backtrack(error_position!(
&b"\\"[..],
ErrorKind::EscapedTransform
)))
);
assert_eq!(
esc(&b"AB\\A"[..]),
Err(ErrMode::Backtrack(error_node_position!(
&b"AB\\A"[..],
ErrorKind::EscapedTransform,
error_position!(&b"A"[..], ErrorKind::Tag)
)))
);
fn esc2(i: &[u8]) -> IResult<&[u8], String> {
map(
escaped_transform(
alpha,
'&',
alt((
value("è".as_bytes(), tag("egrave;")),
value("à".as_bytes(), tag("agrave;")),
)),
),
to_s,
)(i)
}
assert_eq!(
esc2(&b"abèDEF;"[..]),
Ok((&b";"[..], String::from("abèDEF")))
);
assert_eq!(
esc2(&b"abèDàEF;"[..]),
Ok((&b";"[..], String::from("abèDàEF")))
);
}
#[cfg(feature = "std")]
#[test]
fn escape_transform_str() {
use crate::character::complete::alpha1 as alpha;
fn esc(i: &str) -> IResult<&str, String> {
escaped_transform(
alpha,
'\\',
alt((
value("\\", tag("\\")),
value("\"", tag("\"")),
value("\n", tag("n")),
)),
)(i)
}
assert_eq!(esc("abcd;"), Ok((";", String::from("abcd"))));
assert_eq!(esc("ab\\\"cd;"), Ok((";", String::from("ab\"cd"))));
assert_eq!(esc("\\\"abcd;"), Ok((";", String::from("\"abcd"))));
assert_eq!(esc("\\n;"), Ok((";", String::from("\n"))));
assert_eq!(esc("ab\\\"12"), Ok(("12", String::from("ab\""))));
assert_eq!(
esc("AB\\"),
Err(ErrMode::Backtrack(error_position!(
"\\",
ErrorKind::EscapedTransform
)))
);
assert_eq!(
esc("AB\\A"),
Err(ErrMode::Backtrack(error_node_position!(
"AB\\A",
ErrorKind::EscapedTransform,
error_position!("A", ErrorKind::Tag)
)))
);
fn esc2(i: &str) -> IResult<&str, String> {
escaped_transform(
alpha,
'&',
alt((value("è", tag("egrave;")), value("à", tag("agrave;")))),
)(i)
}
assert_eq!(esc2("abèDEF;"), Ok((";", String::from("abèDEF"))));
assert_eq!(
esc2("abèDàEF;"),
Ok((";", String::from("abèDàEF")))
);
fn esc3(i: &str) -> IResult<&str, String> {
escaped_transform(
alpha,
'␛',
alt((value("\0", tag("0")), value("\n", tag("n")))),
)(i)
}
assert_eq!(esc3("a␛0bc␛n"), Ok(("", String::from("a\0bc\n"))));
}
}