Merge branch 'partials'

This commit is contained in:
Tom Alexander 2020-04-07 19:56:44 -04:00
commit ad33f98404
Signed by: talexander
GPG Key ID: D3A179C9A53C0EDE

View File

@ -1,11 +1,11 @@
use nom::branch::alt;
use nom::bytes::complete::escaped;
use nom::bytes::complete::escaped_transform;
use nom::bytes::complete::is_a;
use nom::bytes::complete::is_not;
use nom::bytes::complete::tag;
use nom::bytes::complete::take_until;
use nom::character::complete::one_of;
use nom::character::complete::space1;
use nom::combinator::map;
use nom::combinator::opt;
use nom::combinator::recognize;
@ -15,8 +15,10 @@ use nom::combinator::verify;
use nom::multi::many0;
use nom::multi::many1;
use nom::multi::separated_list;
use nom::multi::separated_nonempty_list;
use nom::sequence::delimited;
use nom::sequence::preceded;
use nom::sequence::separated_pair;
use nom::sequence::tuple;
use nom::IResult;
@ -30,6 +32,7 @@ enum DustTag<'a> {
DTNotExists(Container<'a>),
DTBlock(NamedBlock<'a>),
DTInlinePartial(NamedBlock<'a>),
DTPartial(ParameterizedBlock<'a>),
}
#[derive(Clone, Debug, PartialEq)]
@ -86,6 +89,24 @@ struct NamedBlock<'a> {
contents: Option<Body<'a>>,
}
#[derive(Clone, Debug, PartialEq)]
struct ParameterizedBlock<'a> {
name: String,
params: Vec<KVPair<'a>>,
}
#[derive(Clone, Debug, PartialEq)]
enum RValue<'a> {
RVPath(Path<'a>),
RVString(String),
}
#[derive(Clone, Debug, PartialEq)]
struct KVPair<'a> {
key: &'a str,
value: RValue<'a>,
}
#[derive(Clone, Debug, PartialEq)]
struct Body<'a> {
elements: Vec<TemplateElement<'a>>,
@ -115,6 +136,7 @@ fn dust_tag(i: &str) -> IResult<&str, DustTag> {
conditional("{^", DustTag::DTNotExists),
named_block("{+", DustTag::DTBlock),
named_block("{<", DustTag::DTInlinePartial),
parameterized_self_closing_block("{>", DustTag::DTPartial),
))(i)
}
@ -152,7 +174,25 @@ fn key(i: &str) -> IResult<&str, &str> {
/// A series of keys separated by '.' to reference a variable in the context
fn path(i: &str) -> IResult<&str, Path> {
map(separated_list(tag("."), key), |body| Path { keys: body })(i)
map(separated_nonempty_list(tag("."), key), |body| Path {
keys: body,
})(i)
}
/// Either a literal or a path to a value
fn rvalue(i: &str) -> IResult<&str, RValue> {
alt((
map(path, RValue::RVPath),
map(quoted_string, RValue::RVString),
))(i)
}
/// Parameters for a partial
fn key_value_pair(i: &str) -> IResult<&str, KVPair> {
map(separated_pair(key, tag("="), rvalue), |(k, v)| KVPair {
key: k,
value: v,
})(i)
}
/// Display a value from the context
@ -290,6 +330,33 @@ where
}
}
fn parameterized_self_closing_block<'a, F>(
open_matcher: &'static str,
constructor: F,
) -> impl Fn(&'a str) -> IResult<&'a str, DustTag<'a>>
where
F: Fn(ParameterizedBlock<'a>) -> DustTag<'a>,
{
move |i: &'a str| {
let (i, (name, params)) = delimited(
tag(open_matcher),
tuple((
alt((map(key, String::from), quoted_string)),
opt(preceded(space1, separated_list(space1, key_value_pair))),
)),
tag("/}"),
)(i)?;
Ok((
i,
constructor(ParameterizedBlock {
name: name,
params: params.unwrap_or(Vec::new()),
}),
))
}
}
fn filter(i: &str) -> IResult<&str, Filter> {
preceded(
tag("|"),
@ -620,4 +687,50 @@ mod tests {
Ok(("", r#"foo"bar\baz"#.to_owned()))
);
}
#[test]
fn test_unquoted_partial() {
assert_eq!(
dust_tag(r#"{>foo bar=baz animal="cat"/}"#),
Ok((
"",
DustTag::DTPartial(ParameterizedBlock {
name: "foo".to_owned(),
params: vec![
KVPair {
key: "bar",
value: RValue::RVPath(Path { keys: vec!["baz"] })
},
KVPair {
key: "animal",
value: RValue::RVString("cat".to_owned())
}
]
})
))
);
}
#[test]
fn test_quoted_partial() {
assert_eq!(
dust_tag(r#"{>"template name * with * special \" characters" bar=baz animal="cat"/}"#),
Ok((
"",
DustTag::DTPartial(ParameterizedBlock {
name: r#"template name * with * special " characters"#.to_owned(),
params: vec![
KVPair {
key: "bar",
value: RValue::RVPath(Path { keys: vec!["baz"] })
},
KVPair {
key: "animal",
value: RValue::RVString("cat".to_owned())
}
]
})
))
);
}
}