aboutsummaryrefslogtreecommitdiff
path: root/core/src/test/java/coffee/liz/lambda/eval/InterpreterTest.java
blob: 4b6378294b4f8cee502ebbcda5c309951e31f30f (plain) (blame)
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
92
93
94
95
96
97
98
99
100
101
102
103
package coffee.liz.lambda.eval;

import coffee.liz.lambda.LambdaDriver;
import coffee.liz.lambda.ast.Expression.IdentifierExpression;
import coffee.liz.lambda.ast.LambdaProgram;
import coffee.liz.lambda.ast.SourceCode;
import coffee.liz.lambda.bind.Tick;
import coffee.liz.lambda.bind.ToChurch;
import org.junit.jupiter.api.Test;

import java.util.List;

import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertInstanceOf;
import static org.junit.jupiter.api.Assertions.assertThrows;

class InterpreterTest {

	@Test
	public void identity() {
		final Value result = LambdaDriver.interpret(SourceCode.ofArrow("x -> x"));

		final Value.Closure closure = assertInstanceOf(Value.Closure.class, result);
		assertEquals("x", closure.parameter());
		assertInstanceOf(IdentifierExpression.class, closure.body());
	}

	@Test
	public void identityApplication() {
		final Value result = LambdaDriver.interpret(SourceCode.ofArrow("(x -> x)(y)"));

		assertEquals(new Value.Free("y"), result);
	}

	@Test
	public void nestedApplication() {
		final Value result = LambdaDriver.interpret(SourceCode.ofArrow("(f -> g -> x -> f(g)(x))(x -> x)(y -> y)(x)"));

		assertEquals(new Value.Free("x"), result);
	}

	@Test
	public void cons() {
		final Value result = LambdaDriver.interpret(SourceCode.ofArrow("""
				let pair = a -> b -> f -> f(a)(b);
				let second = x -> y -> y;
				pair(x)(y)(second)
				"""));

		assertEquals(new Value.Free("y"), result);
	}

	@Test
	public void fibonacci() {
		final String source = """
				let true = t -> f -> t;
				let false = t -> f -> f;

				let pair = x -> y -> f -> f(x)(y);
				let fst = p -> p(x -> y -> x);
				let snd = p -> p(x -> y -> y);

				let 0 = f -> x -> x;
				let 1 = f -> x -> f(x);

				let succ = n -> f -> x -> f(n(f)(x));
				let plus = m -> n -> f -> x -> m(f)(n(f)(x));
				let next = p -> pair(snd(p))(succ(snd(p)));
				let pred = n -> fst(n(next)(pair(0)(0)));

				let iszero = n -> n(x -> false)(true);
				let isone = n -> iszero(pred(n));

				let y = f -> (x -> f(x(x)))(x -> f(x(x)));

				let fib = y(fib -> n ->
				    iszero(n) (0)
				      (isone(n) (1)
				        (plus
				          (fib(pred(n)))
				          (fib(pred(pred(n)))))));

				fib(ToChurch(13))(Tick)(dummy)
				""";

		final Tick ticker = new Tick();
		final Value result = LambdaDriver.interpret(SourceCode.ofArrow(source), List.of(ticker, new ToChurch()));

		assertEquals(new Value.Free("dummy"), result);
		assertEquals(233, ticker.getCounter().get());
	}

	@Test
	public void omegaCombinatorThrowsDepthExceeded() {
		final LambdaProgram program = LambdaDriver.parse(SourceCode.ofArrow("(x -> x(x))(x -> x(x))"));
		final Environment env = Environment.from(program.macros(), List.of());

		final EvaluationDepthExceededException exception = assertThrows(EvaluationDepthExceededException.class,
				() -> NormalOrderEvaluator.evaluate(program.expression(), env, 100));

		assertEquals(100, exception.getMaxDepth());
	}
}