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 79 80 81 82 83 84 85 86 87 88 89 90
#![allow(clippy::tabs_in_doc_comments)]
#![warn(rust_2018_idioms, unreachable_pub)]
#![forbid(elided_lifetimes_in_paths, unsafe_code)]
//! This crate defines three parsing combinators for the [chumsky parsing library](chumsky):
//!
//! - [`not_starting_with`]: This combinator takes a list of patterns, and
//! matches the shortest string from the input that diverges from all
//! patterns.
//! - [`not_containing`]: This combinator takes a list of patterns, and
//! any string that does not contain any of the patterns.
//! - [`branch`]: This combinator allows branching into a parser. Each
//! branch defines two parsers. When the first parser matches, it
//! chooses that branch and that branch only, even if the second parser
//! fails. The second parser is then used to produce the output type.
//! You can combine as many branches as you want (similar to `if else`).
//! Then, you have to define an else branch which just takes a `String`
//! and needs to produce output from that. Useful if you want to parse
//! verbatim input plus some syntax.
//!
//! # Example
//!
//! ```rust
//! use chumsky::prelude::*;
//! use chumsky_branch::prelude::*;
//!
//! #[derive(Debug, Eq, PartialEq)]
//! enum Token {
//! Placeholder(String),
//! Comment(String),
//! Verbatim(String)
//! }
//!
//! impl Token {
//! fn lexer() -> impl Parser<char, Self, Error = Simple<char>> {
//! branch(
//! "{{",
//! text::ident().then_ignore(just("}}")).map(Self::Placeholder)
//! )
//! .or_branch(
//! "/*",
//! not_containing(["*/"])
//! .then_ignore(just("*/"))
//! .map(Self::Comment)
//! )
//! .or_else(Self::Verbatim)
//! }
//! }
//!
//! fn lexer() -> impl Parser<char, Vec<Token>, Error = Simple<char>> {
//! Token::lexer().repeated().then_ignore(end())
//! }
//!
//! let input = "/* Greet the user */Hello {{name}}!";
//! assert_eq!(&lexer().parse(input).unwrap(), &[
//! Token::Comment(" Greet the user ".to_owned()),
//! Token::Verbatim("Hello ".to_owned()),
//! Token::Placeholder("name".to_owned()),
//! Token::Verbatim("!".to_owned())
//! ]);
//! ```
#[cfg(test)]
macro_rules! assert_matches {
($expr:expr, $pat:pat) => {
let expr = $expr;
assert!(
matches!(expr, $pat),
"Assertion failed: expected {} to match {}, but found {:?}",
stringify!($expr),
stringify!($pat),
expr
);
};
}
mod branch;
mod not_containing;
mod not_starting_with;
pub use branch::{branch, Branch};
pub use not_containing::not_containing;
pub use not_starting_with::not_starting_with;
pub mod prelude {
pub use crate::{
branch::branch, not_containing::not_containing,
not_starting_with::not_starting_with
};
}