#nom #recursion #left #parser #handle #extension

nom-recursive

Extension of nom to handle left recursion

6 releases (breaking)

0.5.1 Mar 28, 2024
0.5.0 Mar 23, 2023
0.4.0 Aug 31, 2021
0.3.0 Nov 26, 2020
0.1.1 Aug 15, 2019

#33 in Parser tooling

Download history 565/week @ 2024-08-21 473/week @ 2024-08-28 441/week @ 2024-09-04 572/week @ 2024-09-11 476/week @ 2024-09-18 538/week @ 2024-09-25 711/week @ 2024-10-02 503/week @ 2024-10-09 480/week @ 2024-10-16 538/week @ 2024-10-23 533/week @ 2024-10-30 492/week @ 2024-11-06 584/week @ 2024-11-13 685/week @ 2024-11-20 554/week @ 2024-11-27 576/week @ 2024-12-04

2,462 downloads per month
Used in 11 crates (2 directly)

MIT/Apache

9KB
101 lines

nom-recursive

Extension of nom to handle left recursion.

Actions Status Crates.io Docs.rs

Requirement

nom must be 5.0.0 or later. nom-recursive can be applied to function-style parser only.

The input type of nom parser must implement HasRecursiveInfo trait. Therefore &str and &[u8] can't be used. You can define a wrapper type of &str or &[u8] and implement HasRecursiveInfo.

Alternatively you can use nom_locate::LocatedSpan<T, RecursiveInfo>. This implements HasRecursiveInfo in this crate.

Usage

[dependencies]
nom-recursive = "0.5.1"

Example

use nom::branch::*;
use nom::character::complete::*;
use nom::IResult;
use nom_locate::LocatedSpan;
use nom_recursive::{recursive_parser, RecursiveInfo};

// Input type must implement trait HasRecursiveInfo
// nom_locate::LocatedSpan<T, RecursiveInfo> implements it.
type Span<'a> = LocatedSpan<&'a str, RecursiveInfo>;

pub fn expr(s: Span) -> IResult<Span, String> {
    alt((expr_binary, term))(s)
}

// Apply recursive_parser by custom attribute
#[recursive_parser]
pub fn expr_binary(s: Span) -> IResult<Span, String> {
    let (s, x) = expr(s)?;
    let (s, y) = char('+')(s)?;
    let (s, z) = expr(s)?;
    let ret = format!("{}{}{}", x, y, z);
    Ok((s, ret))
}

pub fn term(s: Span) -> IResult<Span, String> {
    let (s, x) = char('1')(s)?;
    Ok((s, x.to_string()))
}

fn main() {
    let ret = expr(LocatedSpan::new_extra("1+1", RecursiveInfo::new()));
    println!("{:?}", ret.unwrap().1);
}

License

Licensed under either of

at your option.

Contribution

Unless you explicitly state otherwise, any contribution intentionally submitted for inclusion in the work by you, as defined in the Apache-2.0 license, shall be dual licensed as above, without any additional terms or conditions.

Dependencies

~2.5MB
~58K SLoC