summaryrefslogtreecommitdiff
path: root/dhall/src/parser.rs
blob: 057fce2c9ec9faf81fe0ac6432fc06d2e3c30c3b (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
use lalrpop_util;

use crate::grammar;
use crate::grammar_util::BoxExpr;
use crate::lexer::{Lexer, LexicalError, Tok};
use crate::core::{bx, Expr};

pub type ParseError<'i> = lalrpop_util::ParseError<usize, Tok<'i>, LexicalError>;

pub fn parse_expr(s: &str) -> Result<BoxExpr, ParseError>  {
    grammar::ExprParser::new().parse(Lexer::new(s))
}

use pest::Parser;
use pest::error::Error;
use pest::iterators::Pair;
use crate::generated_parser::{DhallParser, Rule};

fn debug_pair(pair: Pair<Rule>) {
    fn aux(indent: usize, pair: Pair<Rule>) {
        let indent_str = "| ".repeat(indent);
        println!(r#"{}{:?}: "{}""#, indent_str, pair.as_rule(), pair.as_str());
        for p in pair.into_inner() {
            aux(indent+1, p);
        }
    }
    aux(0, pair)
}

pub fn parse_expr_pest(s: &str) -> Result<BoxExpr, Error<Rule>>  {
    let parsed_expr = DhallParser::parse(Rule::complete_expression, s)?.next().unwrap();
    debug_pair(parsed_expr.clone());
    // println!("{}", parsed_expr.clone());

    fn parse_pair(pair: Pair<Rule>) -> BoxExpr {
        match pair.as_rule() {
            Rule::natural_literal => bx(Expr::NaturalLit(str::parse(pair.as_str().trim()).unwrap())),
            Rule::plus_expression => {
                let mut inner = pair.into_inner().map(parse_pair);
                let first_expr = inner.next().unwrap();
                inner.fold(first_expr, |acc, e| bx(Expr::NaturalPlus(acc, e)))
            }
            Rule::times_expression => {
                let mut inner = pair.into_inner().map(parse_pair);
                let first_expr = inner.next().unwrap();
                inner.fold(first_expr, |acc, e| bx(Expr::NaturalTimes(acc, e)))
            }
            r => panic!("{:?}", r),
        }
    }

    Ok(parse_pair(parsed_expr))
}


#[test]
fn test_parse() {
    use crate::core::Expr::*;
    let expr = "((22 + 3) * 10)";
    println!("{:?}", parse_expr(expr));
    match parse_expr_pest(expr) {
        Err(e) => println!("{}", e),
        ok => println!("{:?}", ok),
    }
    assert_eq!(parse_expr_pest(expr).unwrap(), parse_expr(expr).unwrap());
    assert!(false);

    println!("test {:?}", parse_expr("3 + 5 * 10"));
    assert!(parse_expr("22").is_ok());
    assert!(parse_expr("(22)").is_ok());
    assert_eq!(parse_expr("3 + 5 * 10").ok(),
               Some(Box::new(NaturalPlus(Box::new(NaturalLit(3)),
                                Box::new(NaturalTimes(Box::new(NaturalLit(5)),
                                                      Box::new(NaturalLit(10))))))));
    // The original parser is apparently right-associative
    assert_eq!(parse_expr("2 * 3 * 4").ok(),
               Some(Box::new(NaturalTimes(Box::new(NaturalLit(2)),
                                 Box::new(NaturalTimes(Box::new(NaturalLit(3)),
                                                       Box::new(NaturalLit(4))))))));
    assert!(parse_expr("((((22))))").is_ok());
    assert!(parse_expr("((22)").is_err());
    println!("{:?}", parse_expr("\\(b : Bool) -> b == False"));
    assert!(parse_expr("\\(b : Bool) -> b == False").is_ok());
    println!("{:?}", parse_expr("foo.bar"));
    assert!(parse_expr("foo.bar").is_ok());
    assert!(parse_expr("[] : List Bool").is_ok());

    // println!("{:?}", parse_expr("< Left = True | Right : Natural >"));
    // println!("{:?}", parse_expr(r#""bl${42}ah""#));
    // assert!(parse_expr("< Left = True | Right : Natural >").is_ok());
}