-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathparser_example.py
129 lines (100 loc) · 2.92 KB
/
parser_example.py
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
121
122
123
124
125
126
127
128
129
from do_notation import with_do_notation
import itertools
# instance Monad Parser where
# return t = Parser $ \s -> [(t, s)]
# m >>= k = Parser $ \s -> [(x, y) | (u, v) <- parse m s, (x, y) <- parse (k u) v]
class Parser(object):
def __init__(self, parse):
# f :: Iterable -> [(value, Iterable)]
self.parse = parse
def bind(self, f):
return Parser(lambda xs: list(itertools.chain(*[f(u).parse(v) for (u, v) in self.parse(xs)])))
@staticmethod
def mreturn(x):
return Parser(lambda xs: [(x, xs)])
parser_zero = Parser(lambda xs: [])
def parse_either(p1, p2):
def inner(xs):
return list(p1.parse(xs)) + list(p2.parse(xs))
return Parser(inner)
def parse_first(p1, p2):
def inner(xs):
combo = list(p1.parse(xs)) + list(p2.parse(xs))
if len(combo) > 0:
return [combo[0]]
return []
return Parser(inner)
item = Parser(lambda xs: [(xs[0], xs[1:])] if len(xs) > 0 else [])
@with_do_notation
def sat(proposition):
with do(Parser) as p:
c = item
mreturn(c) if proposition(c) else parser_zero
return p
def char(c):
return sat(lambda x: x == c)
digit = sat(lambda x: x in ['0', '1', '2', '3', '4', '5', '6', '7', '8', '9'])
@with_do_notation
def parse_string(string):
if string == "":
return Parser.mreturn("")
if len(string) > 0:
with do(Parser) as x:
x = char(string[0])
parse_string(string[1:])
mreturn(string)
return x
return parser_zero
def many(p):
return parse_first(many1(p), Parser.mreturn(''))
@with_do_notation
def many1(p):
with do(Parser) as p2:
x = p
xs = many(p)
mreturn(x + xs)
return p2
def sepby(p, sep):
return parse_first(sepby1(p, sep), Parser.mreturn(''))
@with_do_notation
def sepby1(p, sep):
with do(Parser) as throwaway_a_sep:
sep
p
with do(Parser) as p2:
x = p
xs = many(throwaway_a_sep)
mreturn(x + xs)
return p2
@with_do_notation
def parser_example():
with do(Parser) as p:
i = item
char('b') # pop a 'b', or fail otherwise
rest = parse_string('cde')
mreturn(i+rest)
# match [('acde', 'fg')]
print p.parse('abcdefg')
# match [('xcde', 'fg')]
print p.parse('xbcdefg')
# no match
print p.parse('xbcfefg')
# match [('aaa', 'bcde')]
print many(char('a')).parse('aaabcde')
# match [('12345', 'abc')]
print sepby1(digit, char(',')).parse('1,2,3,4,5abc')
# match [('0123456789', ',')]
print (sepby1(digit,
char(','))
.parse('0,1,2,3,4,5,6,7,8,9,'))
# RuntimeError: maximum recursion depth exceeded
# print (sepby1(digit,
# char(','))
# .parse('0,1,2,3,4,5,6,7,8,9,' * 100))
parser_example()
# [('acde', 'fg')]
# [('xcde', 'fg')]
# []
# [('aaa', 'bcde')]
# [('12345', 'abc')]
# [('0123456789', ',')]