-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathsequence.v
85 lines (77 loc) · 2.61 KB
/
sequence.v
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
module vom
// Based on https://docs.rs/nom/7.1.3/nom/sequence/index.html
// Matches an object from the first parser and discards it, then gets an object
// from the second parser, and finally matches an object from the third parser and
// discards it.
pub fn delimited(first Fn, second Fn, third Fn) Fn {
return fn [first, second, third] (input string) !(string, string, int) {
x, _, len1 := first(input)!
y, output, len2 := second(x)!
z, _, len3 := third(y)!
return z, output, len1 + len2 + len3
}
}
// Gets an object from the first parser, then gets another object from the second parser.
pub fn pair(first Fn, second Fn) FnMany {
return fn [first, second] (input string) !(string, []string, int) {
a, b, len1 := first(input)!
c, d, len2 := second(a)!
return c, [b, d], len1 + len2
}
}
// Matches an object from the first parser and discards it, then gets an object
// from the second parser.
pub fn preceded(first Fn, second Fn) Fn {
parsers := [first, second]
return fn [parsers] (input string) !(string, string, int) {
first, second := parsers[0], parsers[1]
x, _, len1 := first(input)!
y, output, len2 := second(x)!
return y, output, len1 + len2
}
}
// Gets an object from the first parser, then matches an object from the sep_parser
// and discards it, then gets another object from the second parser.
pub fn separated_pair(first Fn, sep Fn, second Fn) FnMany {
return fn [first, sep, second] (input string) !(string, []string, int) {
mut output := []string{}
x, start, len1 := first(input)!
output << start
y, _, len2 := sep(x)!
z, end, len3 := second(y)!
output << end
return z, output, len1 + len2 + len3
}
}
// Gets an object from the first parser, then matches an object from the second
// parser and discards it.
pub fn terminated(first Fn, second Fn) Fn {
return fn [first, second] (input string) !(string, string, int) {
x, output, len1 := first(input)!
y, _, len2 := second(x)!
return y, output, len1 + len2
}
}
// Applies a tuple of parsers one by one and returns their results as a tuple.
pub fn tuple(parsers []Fn) FnMany {
return fn [parsers] (input string) !(string, []string, int) {
mut temp := input
mut output := []string{}
mut total_len := 0
for parser in parsers {
rest, slice, len := parser(temp)!
output << slice
temp = rest
total_len += len
}
return temp, output, total_len
}
}
// Applies a tuple of parsers one by one and returns their results as a string.
pub fn concat(parsers []Fn) Fn {
return fn [parsers] (input string) !(string, string, int) {
parser := tuple(parsers)
rest, output, len := parser(input)!
return rest, output.join(''), len
}
}