aboutsummaryrefslogtreecommitdiff
path: root/lib/analysers/filters.ml
blob: 6b1d843d8e454cbe4edb6249e7b1746c456de934 (plain)
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
(** Build a fragment of the query match a filter *)

module Path = ImportDataTypes.Path
module Expression = ImportExpression
open StdLabels

(** Add a list of expressions into the group *)
let add_filters :
    conf:ImporterSyntax.t -> Chunk.t -> Path.t Expression.T.t list -> unit =
 fun ~conf group -> function
  | [] -> ()
  | any ->
      let rec f ~conf group = function
        | [] -> ()
        | hd :: [] ->
            Chunk.add_expression ~conf group hd;
            Chunk.add_string group ")"
        | hd :: tl ->
            Chunk.add_expression ~conf group hd;
            Chunk.add_string group ")\nAND (";
            f ~conf group tl
      in
      Chunk.add_string group "(";
      f ~conf group any

type 'a cte_acc = {
  n : int;
  has_previous : bool;
  acc : 'a;
  cte_index : int option;
  latest_expression : Path.t Expression.T.t list;
}

let add_inner : conf:ImporterSyntax.t -> int -> Buffer.t -> unit =
 fun ~conf n b ->
  let name = "filter" ^ string_of_int n in
  (* We use an INNER JOIN here because we want to be sure to get all the rows
     fetched by the CTE *)
  Buffer.add_string b "\nINNER JOIN '";
  Buffer.add_string b name;
  Buffer.add_string b "' ON ";
  Buffer.add_string b name;
  Buffer.add_string b ".id = ";
  Buffer.add_string b conf.source.name;
  Buffer.add_string b ".id\n"

let print :
    conf:ImporterSyntax.t ->
    (Chunk.t * Chunk.t) cte_acc ->
    ImporterSyntax.CTE.t ->
    (Chunk.t * Chunk.t) cte_acc =
 fun ~conf acc cte ->
  let predicates, query = acc.acc in
  let n = acc.n in
  let cte_index =
    match cte.ImporterSyntax.CTE.group with
    | Some expression ->
        begin
          if acc.has_previous then Chunk.add_string query ", "
          else Chunk.add_string query "WITH "
        end;
        Chunk.add_string query "filter";
        Chunk.add_string query (string_of_int n);
        Chunk.add_string query " AS (";
        Chunk.add_string query "SELECT ";
        Chunk.add_string query conf.source.name;
        Chunk.add_string query ".id, ";
        Chunk.add_expression ~conf query expression;
        Chunk.add_string query " AS group_function";
        Chunk.create_from_statement_of_chunck conf query;

        if acc.has_previous then begin
          let previous_name = "filter" ^ string_of_int (n - 1) in
          add_inner ~conf (n - 1) query.Chunk.b;

          Chunk.add_string query "WHERE ";
          Chunk.add_string query previous_name;
          Chunk.add_string query ".group_function"
        end;

        begin
          match cte.ImporterSyntax.CTE.filters with
          | [] -> ()
          | _ ->
              Chunk.add_string query " WHERE ";
              add_filters ~conf query cte.ImporterSyntax.CTE.filters
        end;
        Chunk.add_string query ")\n";
        Some acc.n
    | None ->
        (* Do not add the filters in the CTE (we don’t have any) but in the main
         query *)
        Chunk.add_string predicates "WHERE ";
        add_filters ~conf predicates cte.ImporterSyntax.CTE.filters;
        acc.cte_index
  in
  {
    acc with
    has_previous = true;
    n = acc.n + 1;
    cte_index;
    latest_expression = cte.ImporterSyntax.CTE.filters;
  }

let generate_sql :
    conf:ImporterSyntax.t -> ImporterSyntax.CTE.t list -> Chunk.t -> Chunk.t =
 fun ~conf filters links' ->
  let predicates = Chunk.create () and links = Chunk.create () in
  let eval =
    List.fold_left filters
      ~init:
        {
          n = 0;
          has_previous = false;
          acc = (links, predicates);
          cte_index = None;
          latest_expression = [];
        }
      ~f:(print ~conf)
  in
  match (eval.cte_index, eval.latest_expression) with
  | None, [] -> predicates
  | None, _ ->
      Chunk.add_string links' " ";
      Chunk.append ~head:links' ~tail:links;
      predicates
  | Some n, [] ->
      add_inner ~conf n links'.b;
      Chunk.append ~head:links' ~tail:links;
      Chunk.add_string links' "filter";
      Chunk.add_string links' (string_of_int n);
      Chunk.add_string links' ".group_function";
      predicates
  | Some n, _ ->
      add_inner ~conf n links'.b;
      Chunk.append ~head:links' ~tail:links;
      Chunk.add_string links' " AND filter";
      Chunk.add_string links' (string_of_int n);
      Chunk.add_string links' ".group_function";
      predicates