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
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
|
open StdLabels
type 'a window =
| Min of 'a
| Max of 'a
| Counter
| Previous of 'a
| Sum of 'a
type 'a t =
| Empty
| Expr of 'a t
| Literal of string
| Integer of string
| Path of 'a
| Concat of 'a t list
| Function of string * 'a t list
| Nvl of 'a t list
| Join of string * 'a t list
| Window of ('a t window * 'a t list * 'a t list)
| BOperator of binary_operator * 'a t * 'a t
| GEquality of binary_operator * 'a t * 'a t list
| Function' of funct * 'a t list
and binary_operator =
| Equal
| Different
| Add
| Minus
| Division
| LT
| GT
| And
| Or
and funct =
| Cmp
| Trim
| Upper
let name_of_function = function
| Upper -> "UPPER"
| Trim -> "TRIM"
| Cmp -> "CMP"
let name_of_operator = function
| Equal -> "="
| Different -> "<>"
| Add -> "+"
| Minus -> "-"
| Division -> "/"
| LT -> "<"
| GT -> ">"
| And -> " and "
| Or -> " or "
let name_of_window = function
| Min _ -> "min"
| Max _ -> "max"
| Counter -> "counter"
| Previous _ -> "previous"
| Sum _ -> "sum"
let map_window : f:('a -> 'b) -> 'a window -> 'b window =
fun ~f -> function
| Min t -> Min (f t)
| Max t -> Max (f t)
| Counter -> Counter
| Previous t -> Previous (f t)
| Sum t -> Sum (f t)
(** Extract the kind of the window function from the given name. *)
let window_of_name name opt =
match (name, opt) with
| "min", Some p -> Min p
| "max", Some p -> Max p
| "counter", None -> Counter
| "previous", Some p -> Previous p
| "sum", Some p -> Sum p
| _other -> raise Not_found
let rec cmp : ('a -> 'a -> int) -> 'a t -> 'a t -> int =
fun f e1 e2 ->
match (e1, e2) with
| Empty, Empty -> 0
| Literal l1, Literal l2 -> String.compare l1 l2
| Integer l1, Integer l2 -> String.compare l1 l2
| Path p1, Path p2 -> f p1 p2
| Concat elems1, Concat elems2 | Nvl elems1, Nvl elems2 ->
List.compare ~cmp:(cmp f) elems1 elems2
| Function (n1, elems1), Function (n2, elems2) ->
let name_cmp = String.compare n1 n2 in
if name_cmp = 0 then List.compare ~cmp:(cmp f) elems1 elems2 else name_cmp
| Window (s1, l11, l12), Window (s2, l21, l22) -> (
match compare s1 s2 with
| 0 ->
let l1_cmp = List.compare ~cmp:(cmp f) l11 l21 in
if l1_cmp = 0 then List.compare ~cmp:(cmp f) l12 l22 else l1_cmp
| other -> other)
| BOperator (n1, arg11, arg12), BOperator (n2, arg21, arg22) -> begin
match compare n1 n2 with
| 0 -> begin
match cmp f arg11 arg21 with
| 0 -> cmp f arg12 arg22
| other -> other
end
| other -> other
end
(* Any other case *)
| other1, other2 -> Stdlib.compare other1 other2
let fold_values : f:('b -> 'a -> 'b) -> init:'b -> 'a t -> 'b =
fun ~f ~init expression ->
let rec _f acc = function
| Empty | Literal _ | Integer _ -> acc
| Expr e -> _f acc e
| Path p -> f acc p
| Concat pp | Function' (_, pp) | Function (_, pp) | Nvl pp | Join (_, pp)
-> List.fold_left ~f:(fun acc a -> _f acc a) ~init:acc pp
| Window (window_f, pp1, pp2) ->
(* Each window function can have a distinct parameter first. *)
let acc' =
match window_f with
| Counter -> acc
| Min key | Max key | Previous key | Sum key -> _f acc key
in
let eval1 = List.fold_left ~f:(fun acc a -> _f acc a) ~init:acc' pp1 in
List.fold_left ~f:(fun acc a -> _f acc a) ~init:eval1 pp2
| BOperator (_, arg1, arg2) -> _f (_f acc arg1) arg2
| GEquality (_, arg1, arg2) ->
let eval1 = List.fold_left ~f:(fun acc a -> _f acc a) ~init:acc arg2 in
_f eval1 arg1
in
_f init expression
let map : type a b. f:(a -> b) -> a t -> b t =
fun ~f expression ->
let rec map = function
| Expr e -> Expr (map e)
| Empty -> Empty
| Literal s -> Literal s
| Integer i -> Integer i
| Path p -> Path (f p)
| Concat pp -> Concat (List.map ~f:map pp)
| Function' (name, pp) -> Function' (name, List.map ~f:map pp)
| Function (name, pp) -> Function (name, List.map ~f:map pp)
| Nvl pp -> Nvl (List.map ~f:map pp)
| Join (sep, pp) -> Join (sep, List.map ~f:map pp)
| Window (window_f, pp1, pp2) ->
let w = map_window ~f:map window_f in
Window (w, List.map ~f:map pp1, List.map ~f:map pp2)
| BOperator (n, arg1, arg2) -> BOperator (n, map arg1, map arg2)
| GEquality (n, arg1, args) -> GEquality (n, map arg1, List.map ~f:map args)
in
map expression
|