use crate::raw_symbol_token::RawSymbolToken;
use crate::text::parse_result::{IonParseResult, UpgradeIResult, UpgradeParser};
use nom::branch::alt;
use nom::bytes::streaming::tag;
use nom::combinator::{map, peek, value};
use nom::sequence::{delimited, pair, preceded, terminated};
use nom::{IResult, Parser};
use crate::text::parsers::annotations::parse_annotations;
use crate::text::parsers::comments::whitespace_or_comments;
use crate::text::parsers::string::parse_string;
use crate::text::parsers::symbol::{parse_operator, parse_symbol};
use crate::text::parsers::top_level::top_level_value;
use crate::text::parsers::value::{annotated_container_start, annotated_scalar};
use crate::text::text_value::{AnnotatedTextValue, TextValue};
pub(crate) fn container_start(input: &str) -> IonParseResult<TextValue> {
alt((struct_start, list_start, s_expression_start))(input).upgrade()
}
pub(crate) fn struct_start(input: &str) -> IResult<&str, TextValue> {
value(TextValue::StructStart, tag("{"))(input)
}
pub(crate) fn list_start(input: &str) -> IResult<&str, TextValue> {
value(TextValue::ListStart, tag("["))(input)
}
pub(crate) fn s_expression_start(input: &str) -> IResult<&str, TextValue> {
value(TextValue::SExpStart, tag("("))(input)
}
pub(crate) fn struct_end(input: &str) -> IonParseResult<&str> {
preceded(whitespace_or_comments, tag("}").upgrade())(input)
}
pub(crate) fn list_end(input: &str) -> IonParseResult<&str> {
preceded(whitespace_or_comments, tag("]").upgrade())(input)
}
pub(crate) fn s_expression_end(input: &str) -> IonParseResult<&str> {
preceded(whitespace_or_comments, tag(")").upgrade())(input)
}
pub(crate) fn list_value(input: &str) -> IonParseResult<AnnotatedTextValue> {
alt((
list_scalar,
preceded(whitespace_or_comments, annotated_container_start),
))(input)
}
pub(crate) fn list_scalar(input: &str) -> IonParseResult<AnnotatedTextValue> {
delimited(
whitespace_or_comments,
annotated_scalar,
list_delimiter,
)(input)
}
pub(crate) fn list_delimiter(input: &str) -> IonParseResult<()> {
preceded(
whitespace_or_comments,
alt((tag(",").upgrade(), peek(list_end))),
)
.map(|_| ())
.parse(input)
}
pub(crate) fn list_value_or_end(input: &str) -> IonParseResult<Option<AnnotatedTextValue>> {
map(list_end, |_end_marker| None)
.or(map(list_value, Some))
.parse(input)
}
pub(crate) fn s_expression_value(input: &str) -> IonParseResult<AnnotatedTextValue> {
alt((
s_expression_scalar,
preceded(whitespace_or_comments, annotated_container_start),
))(input)
}
pub(crate) fn s_expression_scalar(input: &str) -> IonParseResult<AnnotatedTextValue> {
preceded(
whitespace_or_comments,
alt((
pair(parse_annotations, parse_operator)
.map(|(annotations, value)| AnnotatedTextValue::new(annotations, value)),
top_level_value,
parse_operator.map(|op| op.without_annotations()),
)),
)(input)
}
pub(crate) fn s_expression_value_or_end(input: &str) -> IonParseResult<Option<AnnotatedTextValue>> {
map(s_expression_end, |_end_marker| None)
.or(map(s_expression_value, Some))
.parse(input)
}
pub(crate) fn s_expression_delimiter(input: &str) -> IonParseResult<()> {
Ok((input, ()))
}
pub(crate) fn struct_field_name(input: &str) -> IonParseResult<RawSymbolToken> {
delimited(
whitespace_or_comments,
parse_string.or(parse_symbol),
pair(whitespace_or_comments, tag(":")),
)
.map(|value| match value {
TextValue::String(text) => RawSymbolToken::Text(text),
TextValue::Symbol(token) => token,
other => unreachable!(
"Struct field names can only be strings or symbols. Found a {:?}",
other
),
})
.parse(input)
}
pub(crate) fn struct_field_value(input: &str) -> IonParseResult<AnnotatedTextValue> {
alt((
struct_field_scalar,
preceded(whitespace_or_comments, annotated_container_start),
))(input)
}
pub(crate) fn struct_field_scalar(input: &str) -> IonParseResult<AnnotatedTextValue> {
terminated(
top_level_value,
struct_delimiter,
)(input)
}
pub(crate) fn struct_field_name_or_end(input: &str) -> IonParseResult<Option<RawSymbolToken>> {
map(struct_end, |_end_marker| None)
.or(map(struct_field_name, Some))
.parse(input)
}
pub(crate) fn struct_delimiter(input: &str) -> IonParseResult<()> {
preceded(whitespace_or_comments, alt((tag(","), peek(struct_end))))
.map(|_| ())
.parse(input)
}
#[cfg(test)]
mod container_parsing_tests {
use rstest::*;
use crate::raw_symbol_token::{local_sid_token, text_token};
use crate::text::parsers::unit_test_support::{parse_test_err, parse_test_ok};
use crate::text::text_value::TextValue;
use crate::types::{Decimal, Int};
use super::*;
#[rstest]
#[case::start_of_struct("{", TextValue::StructStart)]
#[case::start_of_list("[", TextValue::ListStart)]
#[case::start_of_s_expression("(", TextValue::SExpStart)]
fn test_parse_container_start_ok(#[case] text: &str, #[case] expected: TextValue) {
parse_test_ok(container_start, text, expected)
}
#[rstest]
#[case("5")]
#[case("true")]
#[case("foo")]
#[case("foo::{")]
#[case("\"hello\"")]
#[case("<")]
fn test_parse_container_start_err(#[case] text: &str) {
parse_test_err(container_start, text)
}
#[rstest]
#[case("5,", TextValue::Int(Int::I64(5)).without_annotations())]
#[case("foo::bar::5,", TextValue::Int(Int::I64(5)).with_annotations(["foo", "bar"]))]
#[case("foo::bar,", TextValue::Symbol(text_token("bar")).with_annotations("foo"))]
#[case("bar]", TextValue::Symbol(text_token("bar")).without_annotations())]
#[case("7.]", TextValue::Decimal(Decimal::new(7, 0)).without_annotations())]
#[should_panic]
#[case("5 ", TextValue::String(String::from("<should panic>")).without_annotations())]
#[should_panic]
#[case(", ", TextValue::String(String::from("<should panic>")).without_annotations())]
fn test_parse_list_values(#[case] text: &str, #[case] expected: AnnotatedTextValue) {
parse_test_ok(list_value, text, expected);
}
#[rstest]
#[case("'++',", Some(TextValue::Symbol(text_token("++")).without_annotations()))]
#[case("foo::'++',", Some(TextValue::Symbol(text_token("++")).with_annotations("foo")))]
#[case("5 ,", Some(TextValue::Int(Int::I64(5)).without_annotations()))]
#[case("5]", Some(TextValue::Int(Int::I64(5)).without_annotations()))]
#[case("]", None)]
#[case(" ]", None)]
#[case(" /*comment*/ ]", None)]
fn test_parse_list_value_or_end(
#[case] text: &str,
#[case] expected: Option<AnnotatedTextValue>,
) {
parse_test_ok(list_value_or_end, text, expected);
}
#[rstest]
#[case("++ ", TextValue::Symbol(text_token("++")).without_annotations())]
#[case("foo::++ ", TextValue::Symbol(text_token("++")).with_annotations("foo"))]
#[case("5 ", TextValue::Int(Int::I64(5)).without_annotations())]
#[case("5)", TextValue::Int(Int::I64(5)).without_annotations())]
#[case("foo::bar::5 ", TextValue::Int(Int::I64(5)).with_annotations(["foo", "bar"]))]
#[case("foo::bar 0", TextValue::Symbol(text_token("bar")).with_annotations("foo"))]
#[case("bar)", TextValue::Symbol(text_token("bar")).without_annotations())]
#[case("7.)", TextValue::Decimal(Decimal::new(7, 0)).without_annotations())]
#[should_panic]
#[case("5, ", TextValue::String(String::from("<should panic>")).without_annotations())]
#[should_panic]
#[case("5]", TextValue::String(String::from("<should panic>")).without_annotations())]
fn test_parse_s_expression_values(#[case] text: &str, #[case] expected: AnnotatedTextValue) {
parse_test_ok(s_expression_value, text, expected);
}
#[rstest]
#[case("++ ", Some(TextValue::Symbol(text_token("++")).without_annotations()))]
#[case("foo::++ ", Some(TextValue::Symbol(text_token("++")).with_annotations("foo")))]
#[case("5 ", Some(TextValue::Int(Int::I64(5)).without_annotations()))]
#[case(")", None)]
#[case(" )", None)]
#[case(" /*comment*/ )", None)]
fn test_parse_s_expression_value_or_end(
#[case] text: &str,
#[case] expected: Option<AnnotatedTextValue>,
) {
parse_test_ok(s_expression_value_or_end, text, expected);
}
#[rstest]
#[case("5,", TextValue::Int(Int::I64(5)).without_annotations())]
#[case("5 ,", TextValue::Int(Int::I64(5)).without_annotations())]
#[case("foo::bar::5,", TextValue::Int(Int::I64(5)).with_annotations(["foo", "bar"]))]
#[case("foo::bar,", TextValue::Symbol(text_token("bar")).with_annotations("foo"))]
#[case("bar}", TextValue::Symbol(text_token("bar")).without_annotations())]
#[case("7.}", TextValue::Decimal(Decimal::new(7, 0)).without_annotations())]
#[should_panic]
#[case("5 ", TextValue::String(String::from("<should panic>")).without_annotations())]
#[should_panic]
#[case(", ", TextValue::String(String::from("<should panic>")).without_annotations())]
fn test_parse_struct_field_values(#[case] text: &str, #[case] expected: AnnotatedTextValue) {
parse_test_ok(struct_field_value, text, expected);
}
#[rstest]
#[case("foo:", text_token("foo"))]
#[case(" foo :", text_token("foo"))]
#[case(
"/* Here's a field name */ foo // And here's a delimiter\n:",
text_token("foo")
)]
#[case("'foo':", text_token("foo"))]
#[case(" 'foo' :", text_token("foo"))]
#[case("$10:", local_sid_token(10))]
#[case(" $10 :", local_sid_token(10))]
#[case("\"foo\":", text_token("foo"))]
#[case(" \"foo\" :", text_token("foo"))]
fn test_parse_struct_field_name(#[case] text: &str, #[case] expected: RawSymbolToken) {
parse_test_ok(struct_field_name, text, expected);
}
#[rstest]
#[case("foo:", Some(text_token("foo")))]
#[case(" foo :", Some(text_token("foo")))]
#[case("'foo':", Some(text_token("foo")))]
#[case("}", None)]
#[case(" }", None)]
#[case("/*comment*/}", None)]
fn test_parse_struct_field_name_or_end(
#[case] text: &str,
#[case] expected: Option<RawSymbolToken>,
) {
parse_test_ok(struct_field_name_or_end, text, expected);
}
}