diff options
Diffstat (limited to 'lib/syntax/importerSyntax.ml')
-rw-r--r-- | lib/syntax/importerSyntax.ml | 137 |
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 = []; + } |