aboutsummaryrefslogtreecommitdiffstats
path: root/askama_derive/src/parser.rs
diff options
context:
space:
mode:
authorLibravatar Tuomas Siipola <tuomas@zpl.fi>2020-01-12 00:47:52 +0200
committerLibravatar Dirkjan Ochtman <dirkjan@ochtman.nl>2020-01-12 08:15:04 +0100
commit80148aa75335563106abae8680197e4adf3eb2eb (patch)
treed2c593be4240b086caa5557c150d3546f467322b /askama_derive/src/parser.rs
parent100679e69d37355f3683df43e58aab21312ea147 (diff)
downloadaskama-80148aa75335563106abae8680197e4adf3eb2eb.tar.gz
askama-80148aa75335563106abae8680197e4adf3eb2eb.tar.bz2
askama-80148aa75335563106abae8680197e4adf3eb2eb.zip
Support escaping in string literals
Do not attempt to parse escape sequences thoroughly. Instead let the Rust compiler to check the string literals and provide nice error messages if necessary.
Diffstat (limited to '')
-rw-r--r--askama_derive/src/parser.rs50
1 files changed, 30 insertions, 20 deletions
diff --git a/askama_derive/src/parser.rs b/askama_derive/src/parser.rs
index 91c2d6a..910815a 100644
--- a/askama_derive/src/parser.rs
+++ b/askama_derive/src/parser.rs
@@ -1,6 +1,6 @@
use nom::branch::alt;
-use nom::bytes::complete::{is_not, tag, take_until};
-use nom::character::complete::{char, digit1};
+use nom::bytes::complete::{escaped, is_not, tag, take_until};
+use nom::character::complete::{anychar, char, digit1};
use nom::combinator::{complete, map, opt};
use nom::error::ParseError;
use nom::multi::{many0, many1, separated_list, separated_nonempty_list};
@@ -251,40 +251,50 @@ fn param_num_lit(i: &[u8]) -> IResult<&[u8], MatchParameter> {
map(num_lit, |s| MatchParameter::NumLit(s))(i)
}
+fn str_lit(i: &[u8]) -> IResult<&[u8], &str> {
+ map(
+ delimited(
+ char('\"'),
+ opt(escaped(is_not("\\\""), '\\', anychar)),
+ char('\"'),
+ ),
+ |s| s.map(|s| str::from_utf8(s).unwrap()).unwrap_or(""),
+ )(i)
+}
+
fn expr_str_lit(i: &[u8]) -> IResult<&[u8], Expr> {
- map(delimited(char('"'), take_until("\""), char('"')), |s| {
- Expr::StrLit(str::from_utf8(s).unwrap())
- })(i)
+ map(str_lit, |s| Expr::StrLit(s))(i)
}
fn variant_str_lit(i: &[u8]) -> IResult<&[u8], MatchVariant> {
- map(delimited(char('"'), is_not("\""), char('"')), |s| {
- MatchVariant::StrLit(str::from_utf8(s).unwrap())
- })(i)
+ map(str_lit, |s| MatchVariant::StrLit(s))(i)
}
fn param_str_lit(i: &[u8]) -> IResult<&[u8], MatchParameter> {
- map(delimited(char('"'), is_not("\""), char('"')), |s| {
- MatchParameter::StrLit(str::from_utf8(s).unwrap())
- })(i)
+ map(str_lit, |s| MatchParameter::StrLit(s))(i)
+}
+
+fn char_lit(i: &[u8]) -> IResult<&[u8], &str> {
+ map(
+ delimited(
+ char('\''),
+ opt(escaped(is_not("\\\'"), '\\', anychar)),
+ char('\''),
+ ),
+ |s| s.map(|s| str::from_utf8(s).unwrap()).unwrap_or(""),
+ )(i)
}
fn expr_char_lit(i: &[u8]) -> IResult<&[u8], Expr> {
- map(delimited(char('\''), take_until("'"), char('\'')), |s| {
- Expr::CharLit(str::from_utf8(s).unwrap())
- })(i)
+ map(char_lit, |s| Expr::CharLit(s))(i)
}
fn variant_char_lit(i: &[u8]) -> IResult<&[u8], MatchVariant> {
- map(delimited(char('\''), is_not("'"), char('\'')), |s| {
- MatchVariant::CharLit(str::from_utf8(s).unwrap())
- })(i)
+ map(char_lit, |s| MatchVariant::CharLit(s))(i)
}
fn param_char_lit(i: &[u8]) -> IResult<&[u8], MatchParameter> {
- map(delimited(char('\''), is_not("'"), char('\'')), |s| {
- MatchParameter::CharLit(str::from_utf8(s).unwrap())
- })(i)
+ map(char_lit, |s| MatchParameter::CharLit(s))(i)
}
fn expr_var(i: &[u8]) -> IResult<&[u8], Expr> {