textparse

A Rust library to declaratively implement parsers that are based on Packrat Parsing.
Examples
The following code implements a parser for a JSON subset format:
use textparse::{
components::{
Char, Eos, Items, NonEmpty, SkipWhitespaces, StartsWith, StaticStr, Until, While,
},
Parse, ParseError, ParseResult, Parser, Position, Span,
};
#[derive(Clone, Span, Parse)]
struct JsonValue(WithoutWhitespaces<JsonValueInner>);
#[derive(Clone, Span, Parse)]
#[parse(name = "a JSON value")]
enum JsonValueInner {
Null(JsonNull),
String(JsonString),
Number(JsonNumber),
Array(JsonArray),
Object(JsonObject),
}
#[derive(Clone, Span, Parse)]
#[parse(name = "`null`")]
struct JsonNull(StartsWith<Null>);
#[derive(Clone, Span, Parse)]
#[parse(name = "a JSON string")]
struct JsonString(Char<'"'>, Until<Char<'"'>>, Char<'"'>);
#[derive(Clone, Span, Parse)]
#[parse(name = "a JSON number")]
struct JsonNumber(NonEmpty<While<Digit>>);
#[derive(Clone, Span, Parse)]
#[parse(name = "a JSON array")]
struct JsonArray(Char<'['>, Csv<JsonValue>, Char<']'>);
#[derive(Clone, Span, Parse)]
#[parse(name = "a JSON object")]
struct JsonObject(Char<'{'>, Csv<JsonObjectItem>, Char<'}'>);
#[derive(Clone, Span, Parse)]
struct JsonObjectItem(WithoutWhitespaces<JsonString>, Char<':'>, JsonValue);
#[derive(Clone, Span, Parse)]
struct Csv<T>(Items<T, Char<','>>);
#[derive(Clone, Span, Parse)]
struct WithoutWhitespaces<T>(SkipWhitespaces, T, SkipWhitespaces);
struct Null;
impl StaticStr for Null {
fn static_str() -> &'static str {
"null"
}
}
#[derive(Debug, Clone, Span)]
struct Digit(Position, Position);
impl Parse for Digit {
fn parse(parser: &mut Parser) -> ParseResult<Self> {
parser
.peek_char()
.filter(|c| c.is_ascii_digit())
.ok_or(ParseError)?;
let (start, end) = parser.consume_chars(1);
Ok(Self(start, end))
}
}
You can run the above parser via examples/check_json.rs as follows:
$ echo '["foo",null,{"key": "value"}, 123]' | cargo run --example check_json
OK: the input string is a JSON text.
$ echo '["foo" null]' | cargo run --example check_json
Error: expected one of ',', or ']'
--> <STDIN>:1:8
|
1 | ["foo" null]
| ^ expected one of ',', or ']'