Skip to content

Commit

Permalink
Updated changelog, bumped version
Browse files Browse the repository at this point in the history
  • Loading branch information
zesterer committed Dec 16, 2021
1 parent c21c722 commit 21bb953
Show file tree
Hide file tree
Showing 12 changed files with 133 additions and 64 deletions.
43 changes: 43 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,49 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0

### Fixed

# [0.7.0] - 2021-12-16

### Added

- A new [tutorial](tutorial.md) to help new users

- `select` macro, a wrapper over `filter_map` that makes extracting data from specific tokens easy
- `choice` parser, a better alternative to long `or` chains (which sometimes have poor compilation performance)
- `todo` parser, that panics when used (but not when created) (akin to Rust's `todo!` macro, but for parsers)
- `keyword` parser, that parses *exact* identifiers

- `from_str` combinator to allow converting a pattern to a value inline, using `std::str::FromStr`
- `unwrapped` combinator, to automatically unwrap an output value inline
- `rewind` combinator, that allows reverting the input stream on success. It's most useful when requiring that a
pattern is followed by some terminating pattern without the first parser greedily consuming it
- `map_err_with_span` combinator, to allow fetching the span of the input that was parsed by a parser before an error
was encountered

- `or_else` combinator, to allow processing and potentially recovering from a parser error
- `SeparatedBy::at_most` to require that a separated pattern appear at most a specific number of times
- `SeparatedBy::exactly` to require that a separated pattern be repeated exactly a specific number of times
- `Repeated::exactly` to require that a pattern be repeated exactly a specific number of times

- More trait implementations for various things, making the crate more useful

### Changed

- Made `just`, `one_of`, and `none_of` significant more useful. They can now accept strings, arrays, slices, vectors,
sets, or just single tokens as before
- Added the return type of each parser to its documentation
- More explicit documentation of parser behaviour
- More doc examples
- Deprecated `seq` (`just` has been generalised and can now be used to parse specific input sequences)
- Sealed the `Character` trait so that future changes are not breaking
- Sealed the `Chain` trait and made it more powerful
- Moved trait constraints on `Parser` to where clauses for improved readability

### Fixed

- Fixed a subtle bug that allowed `separated_by` to parse an extra trailing separator when it shouldn't
- Filled a 'hole' in the `Error` trait's API that conflated a lack of expected tokens with expectation of end of input
- Made recursive parsers use weak reference-counting to avoid memory leaks

# [0.6.0] - 2021-11-22

### Added
Expand Down
2 changes: 1 addition & 1 deletion Cargo.toml
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
[package]
name = "chumsky"
version = "0.6.0"
version = "0.7.0"
description = "A parser library for humans with powerful error recovery"
authors = ["Joshua Barretto <[email protected]>"]
repository = "https://github.com/zesterer/chumsky"
Expand Down
20 changes: 11 additions & 9 deletions examples/brainfuck.rs
Original file line number Diff line number Diff line change
Expand Up @@ -22,18 +22,20 @@ enum Instr {

fn parser() -> impl Parser<char, Vec<Instr>, Error = Simple<char>> {
use Instr::*;
recursive(|bf| choice((
just('<').to(Left),
just('>').to(Right),
just('+').to(Incr),
just('-').to(Decr),
just(',').to(Read),
just('.').to(Write),
))
recursive(|bf| {
choice((
just('<').to(Left),
just('>').to(Right),
just('+').to(Incr),
just('-').to(Decr),
just(',').to(Read),
just('.').to(Write),
))
.or(bf.delimited_by('[', ']').map(Loop))
.recover_with(nested_delimiters('[', ']', [], |_| Invalid))
.recover_with(skip_then_retry_until([']']))
.repeated())
.repeated()
})
.then_ignore(end())
}

Expand Down
2 changes: 1 addition & 1 deletion examples/json.rs
Original file line number Diff line number Diff line change
Expand Up @@ -116,7 +116,7 @@ fn main() {
e.expected()
.map(|expected| match expected {
Some(expected) => expected.to_string(),
None => "end of input".to_string()
None => "end of input".to_string(),
})
.collect::<Vec<_>>()
.join(", ")
Expand Down
2 changes: 1 addition & 1 deletion examples/nano_rust.rs
Original file line number Diff line number Diff line change
Expand Up @@ -614,7 +614,7 @@ fn main() {
e.expected()
.map(|expected| match expected {
Some(expected) => expected.to_string(),
None => "end of input".to_string()
None => "end of input".to_string(),
})
.collect::<Vec<_>>()
.join(", ")
Expand Down
1 change: 0 additions & 1 deletion src/chain.rs
Original file line number Diff line number Diff line change
@@ -1,4 +1,3 @@

mod private {
pub trait Sealed<T> {}

Expand Down
38 changes: 20 additions & 18 deletions src/combinator.rs
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,8 @@ impl<I: Clone, O, A: Parser<I, O, Error = E>, B: Parser<I, O, Error = E>, E: Err
let a_state = stream.save();

// If the first parser succeeded and produced no secondary errors, don't bother trying the second parser
// TODO: Perhaps we should *alwaus* take this route, even if recoverable errors did occur? Seems like an
// inconsistent application of PEG rules...
if a_res.0.is_empty() {
if let (a_errors, Ok(a_out)) = a_res {
return (a_errors, Ok(a_out));
Expand Down Expand Up @@ -428,7 +430,7 @@ impl<A, B, U> SeparatedBy<A, B, U> {
///
/// ```
/// # use chumsky::prelude::*;
/// let r#enum = seq::<_, _, Simple<char>>("enum".chars())
/// let r#enum = text::keyword::<_, _, Simple<char>>("enum")
/// .padded()
/// .ignore_then(text::ident()
/// .padded()
Expand Down Expand Up @@ -987,8 +989,9 @@ impl<I: Clone, O, A: Parser<I, O, Error = E>, F: Fn(E) -> E, E: Error<I>> Parser
let (errors, res) = debugger.invoke(&self.0, stream);
let mapper = |e: Located<I, E>| e.map(&self.1);
(
errors,//errors.into_iter().map(mapper).collect(),
res/*.map(|(out, alt)| (out, alt.map(mapper)))*/.map_err(mapper),
errors, //errors.into_iter().map(mapper).collect(),
res /*.map(|(out, alt)| (out, alt.map(mapper)))*/
.map_err(mapper),
)
}

Expand Down Expand Up @@ -1025,14 +1028,14 @@ impl<I: Clone, O, A: Parser<I, O, Error = E>, F: Fn(E, E::Span) -> E, E: Error<I
let mapper = |e: Located<I, E>| {
let at = e.at;
e.map(|e| {
let span = stream.attempt(|stream| { stream.revert(at); (false, stream.span_since(start)) });
let span = stream.attempt(|stream| {
stream.revert(at);
(false, stream.span_since(start))
});
(self.1)(e, span)
})
};
(
errors,
res.map_err(mapper),
)
(errors, res.map_err(mapper))
}

#[inline]
Expand Down Expand Up @@ -1103,13 +1106,8 @@ impl<
#[derive(Copy, Clone)]
pub struct OrElse<A, F>(pub(crate) A, pub(crate) F);

impl<
I: Clone,
O,
A: Parser<I, O, Error = E>,
F: Fn(E) -> Result<O, E>,
E: Error<I>,
> Parser<I, O> for OrElse<A, F>
impl<I: Clone, O, A: Parser<I, O, Error = E>, F: Fn(E) -> Result<O, E>, E: Error<I>> Parser<I, O>
for OrElse<A, F>
{
type Error = E;

Expand All @@ -1125,7 +1123,11 @@ impl<
let res = match res {
Ok(out) => Ok(out),
Err(err) => match (&self.1)(err.error) {
Err(e) => Err(Located { at: err.at, error: e, phantom: PhantomData }),
Err(e) => Err(Located {
at: err.at,
error: e,
phantom: PhantomData,
}),
Ok(out) => Ok((out, None)),
},
};
Expand Down Expand Up @@ -1167,8 +1169,8 @@ impl<I: Clone, O, A: Parser<I, O, Error = E>, L: Into<E::Label> + Clone, E: Erro
/* TODO: Not this? */
/*if e.at > pre_state
{*/
// Only add the label if we committed to this pattern somewhat
e.map(|e| e.with_label(self.1.clone().into()))
// Only add the label if we committed to this pattern somewhat
e.map(|e| e.with_label(self.1.clone().into()))
/*} else {
e
}*/
Expand Down
2 changes: 1 addition & 1 deletion src/error.rs
Original file line number Diff line number Diff line change
Expand Up @@ -296,7 +296,7 @@ impl<I: fmt::Display + Hash, S: Span> fmt::Display for Simple<I, S> {
}

match self.expected.len() {
0 => {},//write!(f, " but end of input was expected")?,
0 => {} //write!(f, " but end of input was expected")?,
1 => write!(
f,
" but {} was expected",
Expand Down
22 changes: 14 additions & 8 deletions src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -60,7 +60,8 @@ pub mod prelude {
pub use super::{
error::{Error as _, Simple},
primitive::{
any, empty, end, filter, filter_map, just, none_of, one_of, seq, take_until, todo, choice,
any, choice, empty, end, filter, filter_map, just, none_of, one_of, seq, take_until,
todo,
},
recovery::{nested_delimiters, skip_then_retry_until, skip_until},
recursive::{recursive, Recursive},
Expand Down Expand Up @@ -346,6 +347,10 @@ pub trait Parser<I: Clone, O> {

/// Map the primary error of this parser to a result. If the result is [`Ok`], the parser succeeds with that value.
///
/// Note that even if the function returns an [`Ok`], the input stream will still be 'stuck' at the input following
/// the input that triggered the error. You'll need to follow uses of this combinator with a parser that resets
/// the input stream to a known-good state (for example, [`take_until`]).
///
/// The output type of this parser is `U`, the [`Ok`] type of the result.
fn or_else<F>(self, f: F) -> OrElse<Self, F>
where
Expand Down Expand Up @@ -879,13 +884,14 @@ pub trait Parser<I: Clone, O> {
/// Apply a fallback recovery strategy to this parser should it fail.
///
/// There is no silver bullet for error recovery, so this function allows you to specify one of several different
/// strategies at the location of your choice.
///
/// Note that for implementation reasons, adding an error recovery strategy can cause a parser to 'over-commit',
/// missing potentially valid alternative parse routes (*TODO: document this and explain why and when it happens*).
/// Rest assured that this case is generally quite rare and only happens for very loose, almost-ambiguous syntax.
/// If you run into cases that you believe should parse but do not, try removing or moving recovery strategies to
/// fix the problem.
/// strategies at the location of your choice. Prefer an error recovery strategy that more precisely mirrors valid
/// syntax where possible to make error recovery more reliable.
///
/// Because chumsky is a [PEG](https://en.m.wikipedia.org/wiki/Parsing_expression_grammar) parser, which always
/// take the first successful parsing route through a grammar, recovering from an error may cause the parser to
/// erroneously miss alternative valid routes through the grammar that do not generate recoverable errors. If you
/// run into cases where valid syntax fails to parse without errors, this might be happening: consider removing
/// error recovery or switching to a more specific error recovery strategy.
///
/// The output type of this parser is `O`, the same as the original parser.
///
Expand Down
55 changes: 35 additions & 20 deletions src/primitive.rs
Original file line number Diff line number Diff line change
Expand Up @@ -120,10 +120,27 @@ pub fn end<E>() -> End<E> {
End(PhantomData)
}

mod private {
pub trait Sealed<T> {}

impl<T> Sealed<T> for T {}
impl Sealed<char> for String {}
impl<'a> Sealed<char> for &'a str {}
impl<'a, T> Sealed<T> for &'a [T] {}
impl<T, const N: usize> Sealed<T> for [T; N] {}
impl<'a, T, const N: usize> Sealed<T> for &'a [T; N] {}
impl<T> Sealed<T> for Vec<T> {}
impl<T> Sealed<T> for std::collections::LinkedList<T> {}
impl<T> Sealed<T> for std::collections::VecDeque<T> {}
impl<T> Sealed<T> for std::collections::HashSet<T> {}
impl<T> Sealed<T> for std::collections::BTreeSet<T> {}
impl<T> Sealed<T> for std::collections::BinaryHeap<T> {}
}

/// A utility trait to abstract over linear container-like things.
///
/// This trait is likely to change in future versions of the crate, so avoid implementing it yourself.
pub trait Container<T> {
pub trait Container<T>: private::Sealed<T> {
/// An iterator over the items within this container, by value.
type Iter: Iterator<Item = T>;
/// Iterate over the elements of the container (using internal iteration because GATs are unstable).
Expand Down Expand Up @@ -375,15 +392,13 @@ impl<I: Clone + PartialEq, C: Container<I>, E: Error<I>> Parser<I, I> for OneOf<
(_, _, Some(tok)) if self.0.get_iter().any(|not| not == tok) => {
(Vec::new(), Ok((tok, None)))
}
(at, span, found) => {
(
Vec::new(),
Err(Located::at(
at,
E::expected_input_found(span, self.0.get_iter().map(Some), found),
)),
)
}
(at, span, found) => (
Vec::new(),
Err(Located::at(
at,
E::expected_input_found(span, self.0.get_iter().map(Some), found),
)),
),
}
}

Expand Down Expand Up @@ -475,15 +490,13 @@ impl<I: Clone + PartialEq, C: Container<I>, E: Error<I>> Parser<I, I> for NoneOf
(_, _, Some(tok)) if self.0.get_iter().all(|not| not != tok) => {
(Vec::new(), Ok((tok, None)))
}
(at, span, found) => {
(
Vec::new(),
Err(Located::at(
at,
E::expected_input_found(span, Vec::new(), found),
)),
)
}
(at, span, found) => (
Vec::new(),
Err(Located::at(
at,
E::expected_input_found(span, Vec::new(), found),
)),
),
}
}

Expand Down Expand Up @@ -836,7 +849,9 @@ pub struct Choice<T, E>(pub(crate) T, pub(crate) PhantomData<E>);

impl<T: Copy, E> Copy for Choice<T, E> {}
impl<T: Clone, E> Clone for Choice<T, E> {
fn clone(&self) -> Self { Self(self.0.clone(), PhantomData) }
fn clone(&self) -> Self {
Self(self.0.clone(), PhantomData)
}
}

macro_rules! impl_for_tuple {
Expand Down
6 changes: 5 additions & 1 deletion src/recovery.rs
Original file line number Diff line number Diff line change
Expand Up @@ -193,7 +193,11 @@ impl<I: Clone + PartialEq, O, F: Fn(E::Span) -> O, E: Error<I>, const N: usize>
),
None => Located::at(
at,
P::Error::expected_input_found(span, Some(Some(self.1.clone())), None),
P::Error::expected_input_found(
span,
Some(Some(self.1.clone())),
None,
),
),
});
}
Expand Down
4 changes: 1 addition & 3 deletions src/recursive.rs
Original file line number Diff line number Diff line change
Expand Up @@ -30,9 +30,7 @@ type OnceParser<'a, I, O, E> = OnceCell<Box<dyn Parser<I, O, Error = E> + 'a>>;
/// [definition](Recursive::define).
///
/// Prefer to use [`recursive()`], which exists as a convenient wrapper around both operations, if possible.
pub struct Recursive<'a, I, O, E: Error<I>>(
RecursiveInner<OnceParser<'a, I, O, E>>,
);
pub struct Recursive<'a, I, O, E: Error<I>>(RecursiveInner<OnceParser<'a, I, O, E>>);

impl<'a, I: Clone, O, E: Error<I>> Recursive<'a, I, O, E> {
fn cell(&self) -> Rc<OnceParser<'a, I, O, E>> {
Expand Down

0 comments on commit 21bb953

Please sign in to comment.