Yap: Yet another (rust) parsing library

API docs

This crate helps you to parse input strings and slices by building on the Iterator interface.

The aim of this crate is to provide transparent, flexible and easy to understand parsing utilities which work on arbitrary slices and strings. Some goals of this library are:

Have a look at the Tokens trait for all of the parsing methods, and examples for each.

Example

``rust use yap::{ // This trait has all of the parsing methods on it: Tokens, // Allows you to use.into_tokens()` on strings and slices, // to get an instance of the above: IntoTokens };

// Step 1: convert our input into something implementing Tokens // ================================================================

let mut tokens = "10 + 2 x 12-4,foobar".into_tokens();

// Step 2: Parse some things from our tokens // =========================================

[derive(PartialEq,Debug)]

enum Op { Plus, Minus, Multiply }

[derive(PartialEq,Debug)]

enum OpOrDigit { Op(Op), Digit(u32) }

// The Tokens trait builds on Iterator, so we get a next method. fn parse_op(t: &mut impl Tokens) -> Option { match t.next()? { '-' => Some(Op::Minus), '+' => Some(Op::Plus), 'x' => Some(Op::Multiply), _ => None } }

// We also get other useful functions.. fn parsedigits(t: &mut impl Tokens) -> Option { let s: String = t .tokenswhile(|c| c.is_digit(10)) .collect(); s.parse().ok() }

// As well as combinator functions like sep_by_all and surrounded_by.. let opordigit = tokens.sepbyall( |t| t.surroundedby( |t| parsedigits(t).map(OpOrDigit::Digit), |t| { t.skiptokenswhile(|c| c.isasciiwhitespace()); } ), |t| parse_op(t).map(OpOrDigit::Op) );

// Now we've parsed our input into OpOrDigits, let's calculate the result.. let mut currentop = Op::Plus; let mut currentdigit = 0; for d in opordigit { match d { OpOrDigit::Op(op) => { currentop = op }, OpOrDigit::Digit(n) => { match currentop { Op::Plus => { currentdigit += n }, Op::Minus => { currentdigit -= n }, Op::Multiply => { currentdigit *= n }, } }, } } asserteq!(current_digit, 140);

// Step 3: do whatever you like with the rest of the input! // ========================================================

// This is available on the concrete type that strings // are converted into (rather than on the Tokens trait): let remaining = tokens.remaining();

assert_eq!(remaining, ",foobar"); ```