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
|
module RegExp ( match, RegExp, ORegExp(..) ) where
import Data.List
type RegExp a = [ORegExp a]
data ORegExp a = Nil
| Literal a
| Xor (RegExp a) (RegExp a)
| Star (RegExp a)
| Plus (RegExp a)
| AnyLiteral
| Bracket [Either a (a,a)]
| NegBracket [Either a (a,a)]
| Start (RegExp a)
| End (RegExp a)
| Times (Int,Int) (RegExp a)
match :: Ord a => RegExp a -> [a] -> Bool
match re s = case re of
[] -> s == []
(Nil:ors) -> match ors s
(Xor or1 or2:ors) -> match (or1 ++ ors) s || match (or2 ++ ors) s
(Literal c:ors) -> case s of
[] -> False
(d:ds) -> if (d == c) then match ors ds else False
(Star r:ors) -> matchstar r ors s
(Plus r:ors) -> matchplus r ors s
(AnyLiteral:ors) -> case s of
[] -> False
(_:ds) -> match ors ds
(Bracket b:ors) -> case s of
[] -> False
(d:ds) -> (matchbracket b d) && match ors ds
(NegBracket b:ors) -> case s of
[] -> False
(d:ds) -> not (matchbracket b d) && match ors ds
(Start _:_) -> failed
(End _:_) -> failed
(Times (n,m) r:ors) -> matchtimes s n m r ors
matchstar :: Ord a => RegExp a -> RegExp a -> [a] -> Bool
matchstar r rgx st = (||)
(match rgx st)
(tryeach (map (\x -> match r x) (inits st)) (tails st) r rgx)
tryeach :: Ord a => [Bool] -> [[a]] -> RegExp a -> RegExp a -> Bool
tryeach [] [] _ _ = False
tryeach (b:bs) (t:ts) r rgx =
(||)
(if b
then
(match rgx t || matchstar r rgx t)
else False)
(tryeach bs ts r rgx)
matchplus :: Ord a => RegExp a -> RegExp a -> [a] -> Bool
matchplus r rgx st = tryeach (map (\x -> match r x) ini) tls r rgx
where
ini = tail (inits st)
tls = tail (tails st)
matchbracket :: Ord a => [Either a (a,a)] -> a -> Bool
matchbracket [] _ = False
matchbracket (Left c:es) d | c == d = True
| c /= d = matchbracket es d
matchbracket (Right (c1,c2):es) d =
(||)
(d >= c1 && d <= c2)
(matchbracket es d)
matchtimes :: Ord a => [a] -> Int -> Int -> RegExp a -> RegExp a -> Bool
matchtimes s n m r rgx | m == 0 = match rgx s
| m > 0 =
tryeachRestricted (m-n) (map (\x -> match mr x) (inits s)) (tails s) r rgx
where
mr = concat (replicate n r)
tryeachRestricted :: Ord a => Int -> [Bool] -> [[a]] -> RegExp a -> RegExp a
-> Bool
tryeachRestricted _ [] [] _ _ = False
tryeachRestricted m (b:bs) (t:ts) r rgx =
(||)
(if b
then
(match rgx t || matchtimes t 1 m r rgx)
else False)
(tryeachRestricted m bs ts r rgx)
|