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
pub mod types;
use chumsky::Parser;
use chumsky::input::Stream;
use chumsky::prelude::*;

use crate::lexer::{Lexer, Token};
pub mod fun;
pub mod util;
use types::*;
use util::*;

use self::fun::Function;

impl<'s> Value<'s> {
    pub fn parse() -> parser![Spanned<Self>] {
        select! {
            Token::Char(x) => Value::Int(x as _),
            Token::Int(x) => Value::Int(x),
            Token::Float(x) => Value::Float(x),
            Token::String(s) => Value::String(s),
        }
        .map_with(spanned!())
        .labelled("value")
    }
}

impl<'s> Expr<'s> {
    pub fn parse() -> parser![Spanned<Expr<'s>>] {
        recursive(|expr| {
            let inline_expr: parser![Spanned<Expr>] =
                Value::parse().map(|x| x.map(Expr::Value));

            let λ = Λ::parse(expr.clone());
            choice((
                inline_expr,
                Function::parse(λ.clone().map(Spanned::unspan()))
                    .map(Expr::Function)
                    .map_with(spanned!()),
                λ.map(|x| x.map(|x| Expr::Value(Value::Lambda(x)))),
            ))
            .labelled("expr")
        })
    }
}

pub fn top<'s>() -> parser![Spanned<Λ<'s>>] {
    Expr::parse()
        .repeated()
        .collect()
        .map(Λ::of)
        .map_with(spanned!())
}

#[test]
fn parse_expr() {
    // parse_s("a ← λ ( +-🍴 )", Expr::parse());
    let src = r#"+↘"#;
    println!(
        "{:?}",
        crate::lexer::lex(src).map(|x| x.0).collect::<Vec<_>>()
    );
    parse_s(src, top());
}

pub fn stream(lexer: Lexer<'_>, len: usize) -> types::Input<'_> {
    Stream::from_iter(lexer).map(SimpleSpan::new((), len..len), |x| x)
}

pub fn code<'s>(x: &'s str) -> types::Input<'s> {
    stream(crate::lexer::lex(x), x.len())
}

pub fn parse_s<'s, T: std::fmt::Debug>(x: &'s str, p: parser![T]) -> T {
    match crate::ui::display(p.parse(code(x)).into_result(), x) {
        Ok(x) => dbg!(x),
        Err(()) => panic!(),
    }
}