rust-parser-combinator/src/choice/mod.rs

152 lines
3.7 KiB
Rust
Raw Normal View History

2022-10-21 16:19:41 -07:00
use crate::parser::{ParseResult, Parser, ParserInput};
2022-10-16 19:21:43 -07:00
2022-10-16 20:07:45 -07:00
pub fn choice2<P1, P2, I, O, E>(parser1: P1, parser2: P2) -> impl Parser<I, O, E>
2022-10-16 19:21:43 -07:00
where
P1: Parser<I, O, E>,
P2: Parser<I, O, E>,
2022-10-21 16:19:41 -07:00
I: ParserInput + Clone,
2022-10-16 19:21:43 -07:00
{
2022-10-22 22:44:36 -07:00
choice((parser1, parser2))
2022-10-16 19:21:43 -07:00
}
2022-10-23 00:25:20 -07:00
pub fn choice<C, I, O, E>(choices: C) -> impl Parser<I, O, E>
2022-10-16 20:07:45 -07:00
where
C: Choice<I, O, E>,
2022-10-21 16:19:41 -07:00
I: ParserInput + Clone,
2022-10-16 20:07:45 -07:00
{
2022-10-23 00:25:20 -07:00
move |input| choices.parse(input)
2022-10-16 20:07:45 -07:00
}
pub trait Choice<I: Clone, O, E> {
fn parse(&self, input: I) -> ParseResult<I, O, E>;
}
impl<I, O, E, P1, P2> Choice<I, O, E> for (P1, P2)
where
P1: Parser<I, O, E>,
P2: Parser<I, O, E>,
2022-10-21 16:19:41 -07:00
I: ParserInput + Clone,
2022-10-16 20:07:45 -07:00
{
fn parse(&self, input: I) -> ParseResult<I, O, E> {
2022-10-20 23:47:06 -07:00
let parsers = vec![&self.0 as &dyn Parser<I, O, E>, &self.1];
choice_loop(input, parsers)
2022-10-16 20:07:45 -07:00
}
}
2022-10-16 20:16:03 -07:00
impl<I, O, E, P1, P2, P3> Choice<I, O, E> for (P1, P2, P3)
where
P1: Parser<I, O, E>,
P2: Parser<I, O, E>,
P3: Parser<I, O, E>,
2022-10-21 16:19:41 -07:00
I: ParserInput + Clone,
2022-10-16 20:16:03 -07:00
{
fn parse(&self, input: I) -> ParseResult<I, O, E> {
2022-10-20 23:47:06 -07:00
let parsers = vec![&self.0 as &dyn Parser<I, O, E>, &self.1, &self.2];
choice_loop(input, parsers)
2022-10-16 20:16:03 -07:00
}
}
impl<I, O, E, P1, P2, P3, P4> Choice<I, O, E> for (P1, P2, P3, P4)
where
P1: Parser<I, O, E>,
P2: Parser<I, O, E>,
P3: Parser<I, O, E>,
P4: Parser<I, O, E>,
2022-10-21 16:19:41 -07:00
I: ParserInput + Clone,
2022-10-16 20:16:03 -07:00
{
fn parse(&self, input: I) -> ParseResult<I, O, E> {
2022-10-20 23:47:06 -07:00
let parsers = vec![&self.0 as &dyn Parser<I, O, E>, &self.1, &self.2, &self.3];
choice_loop(input, parsers)
2022-10-16 20:16:03 -07:00
}
}
impl<I, O, E, P1, P2, P3, P4, P5> Choice<I, O, E> for (P1, P2, P3, P4, P5)
where
P1: Parser<I, O, E>,
P2: Parser<I, O, E>,
P3: Parser<I, O, E>,
P4: Parser<I, O, E>,
P5: Parser<I, O, E>,
2022-10-21 16:19:41 -07:00
I: ParserInput + Clone,
2022-10-16 20:16:03 -07:00
{
fn parse(&self, input: I) -> ParseResult<I, O, E> {
2022-10-20 23:47:06 -07:00
let parsers = vec![
&self.0 as &dyn Parser<I, O, E>,
&self.1,
&self.2,
&self.3,
&self.4,
];
choice_loop(input, parsers)
2022-10-16 20:16:03 -07:00
}
}
2022-10-20 23:39:51 -07:00
impl<I, O, E, P1, P2, P3, P4, P5, P6> Choice<I, O, E> for (P1, P2, P3, P4, P5, P6)
where
P1: Parser<I, O, E>,
P2: Parser<I, O, E>,
P3: Parser<I, O, E>,
P4: Parser<I, O, E>,
P5: Parser<I, O, E>,
P6: Parser<I, O, E>,
2022-10-21 16:19:41 -07:00
I: ParserInput + Clone,
2022-10-20 23:39:51 -07:00
{
fn parse(&self, input: I) -> ParseResult<I, O, E> {
2022-10-20 23:47:06 -07:00
let parsers = vec![
&self.0 as &dyn Parser<I, O, E>,
&self.1,
&self.2,
&self.3,
&self.4,
&self.5,
];
choice_loop(input, parsers)
}
}
2022-10-20 23:39:51 -07:00
2022-10-20 23:47:06 -07:00
fn choice_loop<I, O, E>(input: I, parsers: Vec<&dyn Parser<I, O, E>>) -> ParseResult<I, O, E>
where
2022-10-21 16:19:41 -07:00
I: ParserInput + Clone,
2022-10-20 23:47:06 -07:00
{
//TODO need a more principled way to return an error when no choices work
let mut err = None;
for parser in parsers.iter() {
match parser.parse(input.clone()) {
Ok(result) => return Ok(result),
Err(e) => {
err = Some(e);
2022-10-20 23:39:51 -07:00
}
}
}
2022-10-20 23:47:06 -07:00
Err(err.unwrap())
2022-10-20 23:39:51 -07:00
}
2022-10-16 19:21:43 -07:00
#[cfg(test)]
mod tests {
use super::*;
2022-10-19 19:42:29 -07:00
use crate::combinators::repeated;
2022-10-16 19:21:43 -07:00
use crate::primitives::literal;
#[test]
fn test_choice() {
2022-10-19 19:42:29 -07:00
let p = choice2(
literal("gnostika").to(1),
repeated(literal(" ")).at_least(1).to(2),
);
2022-10-16 19:21:43 -07:00
assert_eq!(p.parse("gnostika twentynine"), Ok((1, " twentynine")));
}
2022-10-20 23:47:06 -07:00
#[test]
fn test_several_choices() {
let p = choice((
literal("a").to(1),
literal("q").to(10),
repeated(literal("chutney")).to(200),
literal("banana").to(10000),
));
assert_eq!(p.parse("q drugs").unwrap(), (10, " drugs"));
}
2022-10-16 19:21:43 -07:00
}