Skip to content

Commit

Permalink
feat(Scheme): add possiblity to add flags to variable.
Browse files Browse the repository at this point in the history
  • Loading branch information
FardaleM committed Oct 7, 2024
1 parent be21845 commit 1dee844
Showing 1 changed file with 35 additions and 6 deletions.
41 changes: 35 additions & 6 deletions lib/common/Scheme.ml
Original file line number Diff line number Diff line change
@@ -1,8 +1,29 @@
type t = {
vars : Variable.t List.t ;
vars : (Variable.t * Type.t) List.t ;
ty : Type.t ;
}

type flags = NonArrow | NonTuple

type var_type = Frozen | Flags of Variable.Flags.t

let parse_var_type =
let open CCParse in
(string "^" >|= fun _ -> Frozen) <|>
(many ((string ">" >|= fun _ -> NonArrow)
<|> (string "*" >|= fun _ -> NonTuple))
>|= fun l -> Flags (List.fold_left (fun flags m ->
match m with
| NonArrow -> Variable.Flags.(set non_arrow flags)
| NonTuple -> Variable.Flags.(set non_tuple flags))
Variable.Flags.empty l))

let parse_var =
let open CCParse in
let* var_type = parse_var_type in
let+ name = U.word in
(name, var_type)

let of_string env str =
let pos = String.find ~sub:". " str in
let bdgs, ty =
Expand All @@ -13,30 +34,38 @@ let of_string env str =
let vars =
if pos = -1 then
String.HMap.values_list bdgs
|> CCList.map (fun var -> (var, Type.frozen_var env var))
else
let str = String.trim @@ String.take pos str in
let vars =
try CCParse.(parse_string_exn @@ sep ~by:space U.word) str
try CCParse.(parse_string_exn @@ sep ~by:space parse_var) str
with CCParse.ParseError _ -> invalid_arg "Schema.of_string"
in
let vars =
try CCList.map (String.HMap.find bdgs) vars
try
CCList.map (fun (name, var_type) ->
let var = String.HMap.find bdgs name in
match var_type with
| Frozen -> (var, Type.frozen_var env var)
| Flags flags ->
let fresh_var = Variable.Gen.gen flags env.var_gen in
(var, Type.var env fresh_var)
) vars
with Not_found -> invalid_arg "Schema.of_string"
in
vars
in
let vars = CCList.sort_uniq ~cmp:Variable.compare vars in
let vars = CCList.sort_uniq ~cmp:(fun (v1, _) (v2, _) -> Variable.compare v1 v2) vars in
{ vars ; ty }

let to_type env t =
let subst =
t.vars
|> CCList.map (fun var -> var, Type.frozen_var env var)
|> Variable.Map.of_list
in
Subst.apply env subst t.ty

let pp ppf t =
Fmt.pf ppf "@[<2>%a.@ %a@]"
Fmt.(list ~sep:sp Variable.pp) t.vars
Fmt.(list ~sep:sp (fun fmt (v, _) -> Variable.pp fmt v)) t.vars
Type.pp t.ty

0 comments on commit 1dee844

Please sign in to comment.