-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathprecedence-parser.scm
executable file
·120 lines (100 loc) · 3.84 KB
/
precedence-parser.scm
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
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
#!/usr/bin/guile \
-e main -s
!#
; A simple order-of-precedence parser
; exception handling
(use-modules (srfi srfi-34))
; +, *, and unary negation
; our tokens, in order of precedence, and the commands they map to
; #\u indicates that we are inside a unary negation
(define precedence '(#\+ #\* #\- #\u))
(define commands '( + * - -))
;; alternatively we can change this from +, *, and unary negation to an EDMSA
;; parser (PEMDAS is a lie!) by changing the lines above to:
;; (define precedence '(#\+ #\- #\* #\/ #\^))
;; (define commands '( + - * / expt))
; helper functions
(define (remove-spaces str)
(string-filter (lambda (char) (not (eqv? char #\space))) str))
; this is where the magic happens
(define (parse-by-precedence precedence commands str)
(if (string=? str "")
(if (eq? (car precedence) #\u)
0
(raise (string-append "Invalid expression:")))
(if (null? precedence)
(let ((num (string->number str)))
(if num
num
(raise (string-append "\"" str "\" is not a lexeme, so this is "
" an invalid expression:"))))
(let ((subtree
(map
(lambda (substr)
(parse-by-precedence (cdr precedence) (cdr commands) substr))
(string-split str (car precedence)))))
(if (null? (cdr subtree))
(car subtree)
(cons (car commands) subtree))))))
(define (eval-by-precedence precedence commands str)
(eval (parse-by-precedence precedence commands str) (interaction-environment)))
(define (display-parse-and-eval-by-precedence precedence commands str)
(let ((parsed (parse-by-precedence precedence commands str)))
(display parsed)
(display " = ")
(display (eval parsed (interaction-environment)))
(newline)))
; read-eval-print loop
(define (repl)
(use-modules (ice-9 readline))
(let ((line (remove-spaces (readline))))
(guard (err [#t (display (string-append err " " line "\n"))])
(display-parse-and-eval-by-precedence precedence commands line)))
(repl))
(define (main args)
(if (and (not (null? (cdr args))) (equal? (car (cdr args)) "-t"))
(test)
(repl)))
; Use SRFI-64 testing framework.
(use-modules (srfi srfi-64))
(define (test)
;; test with EDMSA precedence rules
(test-begin "EDMSA")
(let ((precedence '(#\+ #\- #\* #\/ #\^))
(commands '( + - * / expt)))
(test-equal
'(+ 1 (- (* 2 3) (/ 4 (expt 5 6))))
(parse-by-precedence precedence commands "1+2*3-4/5^6"))
(test-equal
109371/15625
(eval-by-precedence precedence commands "1+2*3-4/5^6"))
(test-equal
'(+ (- (* (/ (expt 7 8) 9) 2) 4) 3)
(parse-by-precedence precedence commands "7^8/9*2-4+3"))
;; left-to-right rule for division is implemented implicitly
(test-equal
'(/ 8 2 2)
(parse-by-precedence precedence commands "8/2/2"))
;; TODO implement right-to-left rule for exponentiation
(test-expect-fail 1)
(test-equal
'(^ 2 (^ 3 4))
(parse-by-precedence precedence commands "2^3^4")))
(test-end "EDMSA")
;; test with EMDAS precedence rules
(test-begin "EMDAS")
(test-equal
'(- 4 (+ 3 (/ (* 2 1) (expt 5 6))))
(parse-by-precedence '(#\- #\+ #\/ #\* #\^) '(- + / * expt) "4-3+2*1/5^6"))
(test-end "EMDAS")
;; test +, -, and unary negation
(test-begin "AMN")
(let ((precedence '(#\+ #\* #\- #\u))
(commands '( + * - -)))
(test-equal -4 (eval-by-precedence precedence commands "-4"))
(test-equal 14 (eval-by-precedence precedence commands "2+3*4"))
(test-equal -10 (eval-by-precedence precedence commands "2+3*-4"))
(test-error (parse-by-precedence precedence commands "2+3*a"))
(test-error (parse-by-precedence precedence commands "2+*3*4")))
(test-end "AMN")
)