organic/src/parser/plain_text.rs

90 lines
2.8 KiB
Rust
Raw Normal View History

use nom::branch::alt;
2023-04-24 20:32:59 -04:00
use nom::bytes::complete::tag;
use nom::character::complete::anychar;
2023-04-24 20:32:59 -04:00
use nom::combinator::map;
use nom::combinator::peek;
2023-04-22 19:46:27 -04:00
use nom::combinator::recognize;
use nom::combinator::verify;
use nom::multi::many_till;
2023-04-22 19:46:27 -04:00
2023-09-03 12:23:18 -04:00
use super::object_parser::any_object_except_plain_text;
use super::org_source::OrgSource;
2023-04-24 20:32:59 -04:00
use super::radio_link::RematchObject;
2023-09-03 12:23:18 -04:00
use super::util::exit_matcher_parser;
use crate::context::parser_with_context;
use crate::context::RefContext;
2023-04-21 18:36:01 -04:00
use crate::error::Res;
2023-09-03 12:23:18 -04:00
use crate::types::Object;
use crate::types::PlainText;
2023-08-10 20:04:59 -04:00
#[cfg_attr(feature = "tracing", tracing::instrument(ret, level = "debug"))]
pub fn plain_text<'r, 's>(
2023-09-02 22:45:46 -04:00
context: RefContext<'_, 'r, 's>,
input: OrgSource<'s>,
) -> Res<OrgSource<'s>, PlainText<'s>> {
let (remaining, source) = recognize(verify(
many_till(
anychar,
peek(alt((
parser_with_context!(exit_matcher_parser)(context),
parser_with_context!(plain_text_end)(context),
))),
),
|(children, _exit_contents)| !children.is_empty(),
))(input)?;
Ok((
remaining,
PlainText {
source: source.into(),
},
))
}
2023-08-10 20:04:59 -04:00
#[cfg_attr(feature = "tracing", tracing::instrument(ret, level = "debug"))]
fn plain_text_end<'r, 's>(
2023-09-02 22:45:46 -04:00
context: RefContext<'_, 'r, 's>,
input: OrgSource<'s>,
) -> Res<OrgSource<'s>, OrgSource<'s>> {
2023-04-22 19:46:27 -04:00
recognize(parser_with_context!(any_object_except_plain_text)(context))(input)
}
2023-07-14 18:04:01 -04:00
impl<'x> RematchObject<'x> for PlainText<'x> {
2023-08-10 20:04:59 -04:00
#[cfg_attr(feature = "tracing", tracing::instrument(ret, level = "debug"))]
2023-04-24 20:32:59 -04:00
fn rematch_object<'r, 's>(
2023-07-14 18:04:01 -04:00
&'x self,
2023-09-02 22:45:46 -04:00
_context: RefContext<'_, 'r, 's>,
input: OrgSource<'s>,
) -> Res<OrgSource<'s>, Object<'s>> {
2023-04-24 20:32:59 -04:00
map(tag(self.source), |s| {
Object::PlainText(PlainText {
source: Into::<&str>::into(s),
})
2023-04-24 20:32:59 -04:00
})(input)
}
}
#[cfg(test)]
mod tests {
use nom::combinator::map;
2023-04-22 19:46:27 -04:00
use super::*;
2023-09-03 12:23:18 -04:00
use crate::context::Context;
use crate::context::ContextElement;
use crate::context::GlobalSettings;
use crate::context::List;
use crate::types::Source;
#[test]
fn plain_text_simple() {
2023-08-24 17:15:24 -04:00
let input = OrgSource::new("foobarbaz");
2023-09-03 12:23:18 -04:00
let global_settings = GlobalSettings::default();
let initial_context = ContextElement::document_context();
let initial_context = Context::new(&global_settings, List::new(&initial_context));
2023-08-24 17:15:24 -04:00
let plain_text_matcher = parser_with_context!(plain_text)(&initial_context);
let (remaining, result) = map(plain_text_matcher, Object::PlainText)(input).unwrap();
2023-08-24 17:15:24 -04:00
assert_eq!(Into::<&str>::into(remaining), "");
assert_eq!(result.get_source(), Into::<&str>::into(input));
}
}