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
	};
}