1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
use std::cell::RefCell;
use std::iter::Iterator;

use crate::core::parser::Parser;
use crate::core::result::ParseResult;

use rusty_parser_derive::ParserHelper;

// RefCell<Parser> wrapper
// this can be combined with BoxedParser, a Box<Parser> wrapper
// for dynamic parser changes
#[derive(Debug, Clone, ParserHelper)]
pub struct RefCelledParser<ParserType, It>
where
    It: Iterator + Clone,
    ParserType: Parser<It>,
{
    parser: RefCell<ParserType>,
    _phantom: std::marker::PhantomData<It>,
}

impl<ParserType, It> RefCelledParser<ParserType, It>
where
    It: Iterator + Clone,
    ParserType: Parser<It>,
{
    pub fn new(parser: ParserType) -> Self {
        Self {
            parser: RefCell::new(parser),
            _phantom: std::marker::PhantomData,
        }
    }
    // get &RefCell<ChildParser>
    pub fn refcelled_parser(&self) -> &RefCell<ParserType> {
        &self.parser
    }
}

impl<ParserType, It> Parser<It> for RefCelledParser<ParserType, It>
where
    It: Iterator + Clone,
    ParserType: Parser<It>,
{
    type Output = <ParserType as Parser<It>>::Output;

    fn parse(&self, it: It) -> ParseResult<Self::Output, It> {
        self.parser.borrow().parse(it)
    }
    fn match_pattern(&self, it: It) -> ParseResult<(), It> {
        self.parser.borrow().match_pattern(it)
    }
}

#[cfg(test)]
mod test {
    use super::*;
    use crate::{core::singlerange::SingleRangeParser, wrapper::boxed::BoxedParser};

    #[test]
    fn success1() {
        let digit_parser = SingleRangeParser::new('0'..='9');
        let boxed = BoxedParser::new(digit_parser);
        let refed = RefCelledParser::new(boxed);

        let str = "123456abcd";

        let res = refed.parse(str.chars());
        assert_eq!(res.output, Some(('1',)));
        let rest: String = res.it.clone().collect();
        assert_eq!(rest, "23456abcd");

        *(refed.parser.borrow_mut()) = BoxedParser::new(SingleRangeParser::new('a'..='z'));
        let res = refed.parse(res.it);
        assert_eq!(res.output, None);
        let rest: String = res.it.clone().collect();
        assert_eq!(rest, "23456abcd");
    }
}