1
/*
2
 * Hurl (https://hurl.dev)
3
 * Copyright (C) 2024 Orange
4
 *
5
 * Licensed under the Apache License, Version 2.0 (the "License");
6
 * you may not use this file except in compliance with the License.
7
 * You may obtain a copy of the License at
8
 *
9
 *          http://www.apache.org/licenses/LICENSE-2.0
10
 *
11
 * Unless required by applicable law or agreed to in writing, software
12
 * distributed under the License is distributed on an "AS IS" BASIS,
13
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14
 * See the License for the specific language governing permissions and
15
 * limitations under the License.
16
 *
17
 */
18
use crate::ast::*;
19
use crate::parser::error::*;
20
use crate::parser::primitives::*;
21
use crate::parser::ParseResult;
22
use crate::reader::Reader;
23

            
24
470790
pub fn parse(reader: &mut Reader) -> ParseResult<Expr> {
25
470790
    try_literal("{{", reader)?;
26
1055
    let space0 = zero_or_more_spaces(reader)?;
27
1055
    let variable = variable_name(reader)?;
28
1050
    let space1 = zero_or_more_spaces(reader)?;
29

            
30
1050
    if try_literal("}}}", reader).is_err() {
31
1050
        literal("}}", reader)?;
32
    }
33

            
34
1050
    Ok(Expr {
35
1050
        space0,
36
1050
        variable,
37
1050
        space1,
38
1050
    })
39
}
40

            
41
1025
pub fn parse2(reader: &mut Reader) -> ParseResult<Expr> {
42
1025
    let space0 = zero_or_more_spaces(reader)?;
43
1025
    let variable = variable_name(reader)?;
44
1025
    let space1 = zero_or_more_spaces(reader)?;
45

            
46
1025
    Ok(Expr {
47
1025
        space0,
48
1025
        variable,
49
1025
        space1,
50
1025
    })
51
}
52

            
53
2080
fn variable_name(reader: &mut Reader) -> ParseResult<Variable> {
54
2080
    let start = reader.cursor();
55
15981
    let name = reader.read_while(|c| c.is_alphanumeric() || c == '_' || c == '-');
56
2080
    if name.is_empty() {
57
5
        return Err(ParseError::new(
58
5
            start.pos,
59
5
            false,
60
5
            ParseErrorKind::TemplateVariable,
61
5
        ));
62
    }
63
2075
    Ok(Variable {
64
2075
        name,
65
2075
        source_info: SourceInfo::new(start.pos, reader.cursor().pos),
66
2075
    })
67
}
68

            
69
#[cfg(test)]
70
mod tests {
71
    use super::*;
72
    use crate::reader::Pos;
73

            
74
    #[test]
75
    fn test_expr() {
76
        let mut reader = Reader::new("{{ name}}");
77
        assert_eq!(
78
            parse(&mut reader).unwrap(),
79
            Expr {
80
                space0: Whitespace {
81
                    value: String::from(" "),
82
                    source_info: SourceInfo::new(Pos::new(1, 3), Pos::new(1, 4)),
83
                },
84
                variable: Variable {
85
                    name: String::from("name"),
86
                    source_info: SourceInfo::new(Pos::new(1, 4), Pos::new(1, 8)),
87
                },
88
                space1: Whitespace {
89
                    value: String::new(),
90
                    source_info: SourceInfo::new(Pos::new(1, 8), Pos::new(1, 8)),
91
                },
92
            }
93
        );
94
    }
95

            
96
    #[test]
97
    fn test_expr_error() {
98
        let mut reader = Reader::new("{{host>}}");
99
        let error = parse(&mut reader).err().unwrap();
100
        assert_eq!(error.pos, Pos { line: 1, column: 7 });
101
        assert_eq!(
102
            error.kind,
103
            ParseErrorKind::Expecting {
104
                value: String::from("}}")
105
            }
106
        );
107
        assert!(!error.recoverable);
108
    }
109

            
110
    #[test]
111
    fn test_expr_error_eof() {
112
        let mut reader = Reader::new("{{host");
113
        let error = parse(&mut reader).err().unwrap();
114
        assert_eq!(error.pos, Pos { line: 1, column: 7 });
115
        assert_eq!(
116
            error.kind,
117
            ParseErrorKind::Expecting {
118
                value: String::from("}}")
119
            }
120
        );
121
        assert!(!error.recoverable);
122
    }
123

            
124
    #[test]
125
    fn test_variable() {
126
        let mut reader = Reader::new("name");
127
        assert_eq!(
128
            variable_name(&mut reader).unwrap(),
129
            Variable {
130
                name: String::from("name"),
131
                source_info: SourceInfo::new(Pos::new(1, 1), Pos::new(1, 5)),
132
            }
133
        );
134

            
135
        let mut reader = Reader::new("my-id");
136
        assert_eq!(
137
            variable_name(&mut reader).unwrap(),
138
            Variable {
139
                name: String::from("my-id"),
140
                source_info: SourceInfo::new(Pos::new(1, 1), Pos::new(1, 6)),
141
            }
142
        );
143
    }
144
}