ocaml/ocamltest/environments.ml

158 lines
5.2 KiB
OCaml

(**************************************************************************)
(* *)
(* OCaml *)
(* *)
(* Sebastien Hinderer, projet Gallium, INRIA Paris *)
(* *)
(* Copyright 2016 Institut National de Recherche en Informatique et *)
(* en Automatique. *)
(* *)
(* All rights reserved. This file is distributed under the terms of *)
(* the GNU Lesser General Public License version 2.1, with the *)
(* special exception on linking described in the file LICENSE. *)
(* *)
(**************************************************************************)
(* Definition of environments, used to pass parameters to tests and actions *)
open Ocamltest_stdlib
module VariableMap = Map.Make (Variables)
type t = string VariableMap.t
let empty = VariableMap.empty
let to_bindings env =
let f variable value lst = (variable, value) :: lst in
VariableMap.fold f env []
let expand_aux env value =
let bindings = to_bindings env in
let f (variable, value) = ((Variables.name_of_variable variable), value) in
let simple_bindings = List.map f bindings in
let subst s = try (List.assoc s simple_bindings) with Not_found -> "" in
let b = Buffer.create 100 in
try Buffer.add_substitute b subst value; Buffer.contents b with _ -> value
let rec expand env value =
let expanded = expand_aux env value in
if expanded=value then value else expand env expanded
let to_system_env env =
let system_env = Array.make (VariableMap.cardinal env) "" in
let i = ref 0 in
let store variable value =
system_env.(!i) <-
Variables.string_of_binding variable (expand env value);
incr i in
VariableMap.iter store env;
system_env
let lookup variable env =
try Some (expand env (VariableMap.find variable env)) with Not_found -> None
let lookup_nonempty variable env = match lookup variable env with
| None -> None
| Some x as t -> if String.words x = [] then None else t
let lookup_as_bool variable env =
match lookup variable env with
| None -> None
| Some "true" -> Some true
| Some _ -> Some false
let safe_lookup variable env = match lookup variable env with
| None -> ""
| Some value -> value
let is_variable_defined variable env =
VariableMap.mem variable env
let add variable value env = VariableMap.add variable value env
let add_if_undefined variable value env =
if VariableMap.mem variable env then env else add variable value env
let append variable appened_value environment =
let previous_value = safe_lookup variable environment in
let new_value = previous_value ^ appened_value in
VariableMap.add variable new_value environment
let remove = VariableMap.remove
let add_bindings bindings env =
let f env (variable, value) = add variable value env in
List.fold_left f env bindings
let from_bindings bindings = add_bindings bindings empty
let dump_assignment log (variable, value) =
Printf.fprintf log "%s = %s\n%!" (Variables.name_of_variable variable) value
let dump log environment =
List.iter (dump_assignment log) (VariableMap.bindings environment)
(* Initializers *)
type kind = Pre | Post
type env_initializer = out_channel -> t -> t
type initializers =
{
pre: (string, env_initializer) Hashtbl.t;
post: (string, env_initializer) Hashtbl.t;
}
let initializers = {pre = Hashtbl.create 10; post = Hashtbl.create 10}
let get_initializers = function
| Pre -> initializers.pre
| Post -> initializers.post
let register_initializer kind name code =
Hashtbl.add (get_initializers kind) name code
let apply_initializer _log _name code env =
code _log env
let initialize kind log env =
let f = apply_initializer log in
Hashtbl.fold f (get_initializers kind) env
(* Modifiers *)
type modifier =
| Include of string
| Add of Variables.t * string
| Append of Variables.t * string
| Remove of Variables.t
type modifiers = modifier list
exception Empty_modifiers_name
exception Modifiers_name_already_registered of string
exception Modifiers_name_not_found of string
let (registered_modifiers : (string, modifiers) Hashtbl.t) = Hashtbl.create 20
let register_modifiers name modifiers =
if name="" then raise Empty_modifiers_name
else if Hashtbl.mem registered_modifiers name
then raise (Modifiers_name_already_registered name)
else Hashtbl.add registered_modifiers name modifiers
let find_modifiers name =
try Hashtbl.find registered_modifiers name
with Not_found -> raise (Modifiers_name_not_found name)
let rec apply_modifier environment = function
| Include modifiers_name ->
apply_modifiers environment (find_modifiers modifiers_name)
| Add (variable, value) -> add variable value environment
| Append (variable, value) -> append variable value environment
| Remove variable -> remove variable environment
and apply_modifiers environment modifiers =
List.fold_left apply_modifier environment modifiers