aboutsummaryrefslogtreecommitdiff
path: root/lib/syntax/importerSyntax.ml
diff options
context:
space:
mode:
Diffstat (limited to 'lib/syntax/importerSyntax.ml')
-rw-r--r--lib/syntax/importerSyntax.ml137
1 files changed, 137 insertions, 0 deletions
diff --git a/lib/syntax/importerSyntax.ml b/lib/syntax/importerSyntax.ml
new file mode 100644
index 0000000..7788613
--- /dev/null
+++ b/lib/syntax/importerSyntax.ml
@@ -0,0 +1,137 @@
+open StdLabels
+module E = ImportExpression.T
+module Table = ImportDataTypes.Table
+module Path = ImportDataTypes.Path
+module CTE = Cte
+
+let toml_of_table Table.{ file; tab; name } =
+ let values = [ ("file", Otoml.string file); ("name", Otoml.string name) ] in
+ let values =
+ match tab with
+ | 1 -> values
+ | tab -> ("tab", Otoml.integer tab) :: values
+ in
+
+ Otoml.table values
+
+module Extern = struct
+ type t = {
+ intern_key : Path.t E.t;
+ target : Table.t;
+ extern_key : Path.column E.t;
+ allow_missing : bool;
+ match_rule : string option;
+ }
+ [@@deriving show, eq]
+ (** Describe a relation beteween two tables *)
+
+ let toml_of_extern extern =
+ let values =
+ [
+ ( "intern_key",
+ Otoml.string
+ @@ ImportExpression.Repr.repr ~top:true Path.show extern.intern_key );
+ ( "extern_key",
+ Otoml.string
+ @@ ImportExpression.Repr.repr ~top:true
+ (fun v -> ":" ^ ImportDataTypes.Path.column_to_string v)
+ extern.extern_key );
+ ("file", Otoml.string extern.target.file);
+ ("allow_missing", Otoml.boolean extern.allow_missing);
+ ]
+ in
+
+ let values =
+ match extern.target.tab with
+ | 1 -> values
+ | tab -> ("tab", Otoml.integer tab) :: values
+ in
+
+ Otoml.table values
+
+ let toml externs =
+ List.map externs ~f:(fun e -> (e.target.name, toml_of_extern e))
+ |> Otoml.table
+end
+
+type t = {
+ version : int;
+ locale : string option;
+ source : Table.t;
+ externals : Extern.t list;
+ columns : Path.t E.t list;
+ filters : Path.t E.t list;
+ sort : Path.t E.t list;
+ uniq : Path.t E.t list;
+}
+
+let repr t =
+ let repr_expression_list l =
+ Otoml.array
+ (List.map l ~f:(fun v ->
+ Otoml.string (ImportExpression.Repr.repr ~top:true Path.show v)))
+ in
+
+ let sheet =
+ Otoml.table
+ [
+ ("columns", repr_expression_list t.columns);
+ ("filters", repr_expression_list t.filters);
+ ("sort", repr_expression_list t.sort);
+ ("uniq", repr_expression_list t.uniq);
+ ]
+ in
+
+ let values =
+ [
+ ("version", Otoml.integer t.version);
+ ("source", toml_of_table t.source);
+ ("externals", Extern.toml t.externals);
+ ("sheet", sheet);
+ ]
+ in
+
+ Otoml.table values
+
+let get_table_for_name : t -> string option -> Table.t =
+ fun conf name ->
+ match name with
+ | None -> conf.source
+ | Some name ->
+ if String.equal name conf.source.name then conf.source
+ else
+ let ext =
+ List.find conf.externals ~f:(fun (ext : Extern.t) ->
+ String.equal name ext.target.name)
+ in
+ ext.target
+
+let root_table : t -> Table.t = fun conf -> conf.source
+
+let get_dependancies_for_table : t -> Table.t -> Extern.t list =
+ fun conf source ->
+ let is_root = source = conf.source in
+
+ List.filter conf.externals ~f:(fun (ext : Extern.t) ->
+ (* Enumerate the intern_key and check the source pointed by each column *)
+ ImportExpression.T.fold_values ext.intern_key ~init:false
+ ~f:(fun acc expr ->
+ if acc then acc
+ else
+ match expr.Path.alias with
+ | Some v -> String.equal v source.name
+ | None -> is_root))
+
+let latest_version = 1
+
+let dummy_conf =
+ {
+ source = { file = ""; tab = 0; name = "" };
+ version = latest_version;
+ locale = Some "C";
+ externals = [];
+ columns = [];
+ filters = [];
+ sort = [];
+ uniq = [];
+ }