tamer: parse::TransitionResult: Move common Transition into Result
This allows the Results to compose and, importantly, is compatible with `?` without having to put in any extra effort. This makes puts the caller in an awkward spot, so I introduced a utility function `result_tup0_invert` for now; we'll see if that stays or evolves differently. DEV-10863main
parent
9d9b1f30a8
commit
bf5da75096
|
@ -169,16 +169,16 @@ impl<S: ParseState> Transition<S> {
|
|||
///
|
||||
/// This allows [`ParseState::parse_token`] to emit a parsed object and
|
||||
/// corresponds to [`ParseStatus::Object`].
|
||||
pub fn with(self, obj: S::Object) -> (Self, ParseStateResult<S>) {
|
||||
(self, Ok(ParseStatus::Object(obj)))
|
||||
pub fn with(self, obj: S::Object) -> TransitionResult<S> {
|
||||
Ok((self, ParseStatus::Object(obj)))
|
||||
}
|
||||
|
||||
/// A state transition indicating that more data is needed before an
|
||||
/// object can be emitted.
|
||||
///
|
||||
/// This corresponds to [`ParseStatus::Incomplete`].
|
||||
pub fn incomplete(self) -> (Self, ParseStateResult<S>) {
|
||||
(self, Ok(ParseStatus::Incomplete))
|
||||
pub fn incomplete(self) -> TransitionResult<S> {
|
||||
Ok((self, ParseStatus::Incomplete))
|
||||
}
|
||||
|
||||
/// A dead state transition.
|
||||
|
@ -186,16 +186,16 @@ impl<S: ParseState> Transition<S> {
|
|||
/// This corresponds to [`ParseStatus::Dead`],
|
||||
/// and a calling parser should use the provided [`Token`] as
|
||||
/// lookahead.
|
||||
pub fn dead(self, tok: S::Token) -> (Self, ParseStateResult<S>) {
|
||||
(self, Ok(ParseStatus::Dead(tok)))
|
||||
pub fn dead(self, tok: S::Token) -> TransitionResult<S> {
|
||||
Ok((self, ParseStatus::Dead(tok)))
|
||||
}
|
||||
|
||||
/// A transition with corresponding error.
|
||||
///
|
||||
/// This indicates a parsing failure.
|
||||
/// The state ought to be suitable for error recovery.
|
||||
pub fn err<E: Into<S::Error>>(self, err: E) -> (Self, ParseStateResult<S>) {
|
||||
(self, Err(err.into()))
|
||||
pub fn err<E: Into<S::Error>>(self, err: E) -> TransitionResult<S> {
|
||||
Err((self, err.into()))
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -204,7 +204,16 @@ impl<S: ParseState> Transition<S> {
|
|||
/// Conceptually,
|
||||
/// imagine the act of a state transition producing data.
|
||||
/// See [`Transition`] for convenience methods for producing this tuple.
|
||||
pub type TransitionResult<S> = (Transition<S>, ParseStateResult<S>);
|
||||
///
|
||||
/// See also [`result_tup0_invert`] if this common [`Transition`] in the
|
||||
/// inner tuple is inconvenient to work with.
|
||||
pub type TransitionResult<S> = Result<
|
||||
(
|
||||
Transition<S>,
|
||||
ParseStatus<<S as ParseState>::Token, <S as ParseState>::Object>,
|
||||
),
|
||||
(Transition<S>, <S as ParseState>::Error),
|
||||
>;
|
||||
|
||||
/// A streaming parser defined by a [`ParseState`] with exclusive
|
||||
/// mutable access to an underlying [`TokenStream`].
|
||||
|
@ -227,6 +236,20 @@ pub struct Parser<S: ParseState, I: TokenStream<S::Token>> {
|
|||
last_span: Option<Span>,
|
||||
}
|
||||
|
||||
/// Invert a [`Result`] containing a tuple for both [`Ok`] and [`Err`]
|
||||
/// variants where first index of the tuple is a common `C`.
|
||||
///
|
||||
/// The result is a tuple with the common type in the first index,
|
||||
/// and the second index containing the remaining [`Result`] data.
|
||||
pub fn result_tup0_invert<C, T, E>(
|
||||
result: Result<(C, T), (C, E)>,
|
||||
) -> (C, Result<T, E>) {
|
||||
match result {
|
||||
Ok((c, t)) => (c, Ok(t)),
|
||||
Err((c, e)) => (c, Err(e)),
|
||||
}
|
||||
}
|
||||
|
||||
impl<S: ParseState, I: TokenStream<S::Token>> Parser<S, I> {
|
||||
/// Indicate that no further parsing will take place using this parser,
|
||||
/// and [`drop`] it.
|
||||
|
@ -276,7 +299,7 @@ impl<S: ParseState, I: TokenStream<S::Token>> Parser<S, I> {
|
|||
|
||||
let result;
|
||||
(Transition(self.state), result) =
|
||||
take(&mut self.state).parse_token(tok);
|
||||
result_tup0_invert(take(&mut self.state).parse_token(tok));
|
||||
|
||||
use ParseStatus::*;
|
||||
match result {
|
||||
|
|
|
@ -121,7 +121,7 @@ mod test {
|
|||
use super::*;
|
||||
use crate::{
|
||||
convert::ExpectInto,
|
||||
parse::{ParseStatus, Parsed},
|
||||
parse::{result_tup0_invert, ParseStatus, Parsed},
|
||||
sym::GlobalSymbolIntern,
|
||||
};
|
||||
|
||||
|
@ -137,12 +137,12 @@ mod test {
|
|||
// There is no state that we can transition to,
|
||||
// and we're in an empty accepting state.
|
||||
assert_eq!(
|
||||
(
|
||||
// Make sure we're in the same state we started in so that
|
||||
// we know we can accommodate recovery token(s).
|
||||
// Make sure we're in the same state we started in so that
|
||||
// we know we can accommodate recovery token(s).
|
||||
Ok((
|
||||
Transition(AttrParseState::default()),
|
||||
Ok(ParseStatus::Dead(tok.clone()))
|
||||
),
|
||||
ParseStatus::Dead(tok.clone())
|
||||
)),
|
||||
sut.parse_token(tok)
|
||||
);
|
||||
}
|
||||
|
@ -175,12 +175,12 @@ mod test {
|
|||
// This token indicates that we're expecting a value to come next in
|
||||
// the token stream.
|
||||
let (Transition(sut), result) =
|
||||
sut.parse_token(XirToken::AttrName(attr, S));
|
||||
result_tup0_invert(sut.parse_token(XirToken::AttrName(attr, S)));
|
||||
assert_eq!(result, Ok(ParseStatus::Incomplete));
|
||||
|
||||
// But we provide something else unexpected.
|
||||
let (Transition(sut), result) =
|
||||
sut.parse_token(XirToken::Close(None, S2));
|
||||
result_tup0_invert(sut.parse_token(XirToken::Close(None, S2)));
|
||||
assert_eq!(
|
||||
result,
|
||||
Err(AttrParseError::AttrValueExpected(
|
||||
|
@ -200,8 +200,9 @@ mod test {
|
|||
// Rather than checking for that state,
|
||||
// let's actually attempt a recovery.
|
||||
let recover = "value".intern();
|
||||
let (Transition(sut), result) =
|
||||
sut.parse_token(XirToken::AttrValue(recover, S2));
|
||||
let (Transition(sut), result) = result_tup0_invert(
|
||||
sut.parse_token(XirToken::AttrValue(recover, S2)),
|
||||
);
|
||||
assert_eq!(
|
||||
result,
|
||||
Ok(ParseStatus::Object(Attr::new(attr, recover, (S, S2)))),
|
||||
|
|
|
@ -218,14 +218,14 @@ where
|
|||
(NodeExpected(stack), tok) => Self::parse_node(stack, tok),
|
||||
|
||||
(AttrExpected(stack, sa), tok) => match sa.parse_token(tok) {
|
||||
(Transition(sa), Ok(Incomplete)) => {
|
||||
Ok((Transition(sa), Incomplete)) => {
|
||||
Transition(AttrExpected(stack, sa)).incomplete()
|
||||
}
|
||||
(Transition(sa), Ok(Obj(attr))) => {
|
||||
Ok((Transition(sa), Obj(attr))) => {
|
||||
Transition(AttrExpected(stack, sa)).with(Object::Attr(attr))
|
||||
}
|
||||
(_, Ok(Dead(lookahead))) => Self::parse_node(stack, lookahead),
|
||||
(Transition(sa), Err(x)) => {
|
||||
Ok((_, Dead(lookahead))) => Self::parse_node(stack, lookahead),
|
||||
Err((Transition(sa), x)) => {
|
||||
Transition(AttrExpected(stack, sa)).err(x)
|
||||
}
|
||||
},
|
||||
|
|
|
@ -542,18 +542,18 @@ impl<SA: StackAttrParseState> ParseState for Stack<SA> {
|
|||
(AttrState(estack, attrs, sa), tok) => {
|
||||
use ParseStatus::*;
|
||||
match sa.parse_token(tok) {
|
||||
(Transition(sa), Ok(Incomplete)) => {
|
||||
Ok((Transition(sa), Incomplete)) => {
|
||||
Transition(AttrState(estack, attrs, sa)).incomplete()
|
||||
}
|
||||
(Transition(sa), Ok(Object(attr))) => {
|
||||
Ok((Transition(sa), Object(attr))) => {
|
||||
Transition(AttrState(estack, attrs.push(attr), sa))
|
||||
.incomplete()
|
||||
}
|
||||
(_, Ok(Dead(lookahead))) => {
|
||||
Ok((_, Dead(lookahead))) => {
|
||||
BuddingElement(estack.consume_attrs(attrs))
|
||||
.parse_token(lookahead)
|
||||
}
|
||||
(Transition(sa), Err(x)) => {
|
||||
Err((Transition(sa), x)) => {
|
||||
Transition(AttrState(estack, attrs, sa)).err(x.into())
|
||||
}
|
||||
}
|
||||
|
|
Loading…
Reference in New Issue