marser 0.1.0

Parser combinator toolkit with matcher-level backtracking and rich error reporting.
use crate::{
    error::{MatcherRunError, error_handler::ErrorHandler},
    input::{InputStream, SliceableInput},
    matcher::{MatchRunner, Matcher, MatcherCombinator, internal::MatcherImpl},
    parser::{BoundValue, Property, capture::MatchResult},
};

/// Matcher wrapper that stores the consumed input slice into a capture property.
#[derive(Clone)]
pub struct SliceBinder<Match, Prop> {
    pub(super) matcher: Match,
    pub(super) property: Prop,
}

impl<Match, Prop> std::fmt::Debug for SliceBinder<Match, Prop>
where
    Match: std::fmt::Debug,
{
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
        f.debug_struct("SliceBinder")
            .field("matcher", &self.matcher)
            .finish()
    }
}

impl<Match, Prop> SliceBinder<Match, Prop> {
    /// See [`bind_slice`].
    pub fn new(matcher: Match, property: Prop) -> Self {
        Self { matcher, property }
    }
}

/// Convenience constructor for [`SliceBinder`].
pub fn bind_slice<Match, Prop>(matcher: Match, property: Prop) -> SliceBinder<Match, Prop> {
    SliceBinder::new(matcher, property)
}

impl<Match, Prop> MatcherCombinator for SliceBinder<Match, Prop> where Match: MatcherCombinator {}

impl<'src, Inp: SliceableInput<'src>, MRes, Match, Prop> MatcherImpl<'src, Inp, MRes>
    for SliceBinder<Match, Prop>
where
    Match: Matcher<'src, Inp, MRes>,
    MRes: MatchResult,
    Prop: Property<Inp::Slice, MRes> + Clone + 'src,
{
    const CAN_MATCH_DIRECTLY: bool = Match::CAN_MATCH_DIRECTLY;
    const HAS_PROPERTY: bool = true;
    const CAN_FAIL: bool = Match::CAN_FAIL;

    #[inline]
    fn match_with_runner<'a, Runner>(
        &'a self,
        runner: &mut Runner,
        error_handler: &mut impl ErrorHandler,
        input: &mut InputStream<'src, Inp>,
    ) -> Result<bool, MatcherRunError>
    where
        Runner: MatchRunner<'a, 'src, Inp, MRes = MRes>,
        'src: 'a,
    {
        let start_pos = input.get_pos();
        if !runner.run_match(&self.matcher, error_handler, input)? {
            return Ok(false);
        }
        let end_pos = input.get_pos();
        let slice = input.slice(start_pos..end_pos);
        let bound: BoundValue<Inp::Slice, _> = self.property.bind_result(slice);
        runner.register_result(bound);
        Ok(true)
    }
}