Source file ppx_deriving_enum.ml
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
open Ppxlib
open Asttypes
open Parsetree
open Ast_helper
open Ppx_deriving.Ast_convenience
let deriver = "enum"
let raise_errorf = Ppx_deriving.raise_errorf
let attr_value context = Attribute.declare "deriving.enum.value" context
Ast_pattern.(single_expr_payload (eint __)) (fun i -> i)
let constr_attr_value = attr_value Attribute.Context.constructor_declaration
let rtag_attr_value = attr_value Attribute.Context.rtag
let mappings_of_type type_decl =
let map acc mappings attr_value x constr_name =
let value =
match Attribute.get attr_value x with
| Some idx -> idx | None -> acc
in
(value + 1, (value, constr_name) :: mappings)
in
let kind, (_, mappings) =
match type_decl.ptype_kind, type_decl.ptype_manifest with
| Ptype_variant constrs, _ ->
`Regular,
List.fold_left (fun (acc, mappings) ({ pcd_name; pcd_args; pcd_attributes; pcd_loc } as constr) ->
if pcd_args <> Pcstr_tuple([]) then
raise_errorf ~loc:pcd_loc
"%s can be derived only for argumentless constructors" deriver;
map acc mappings constr_attr_value constr pcd_name)
(0, []) constrs
| Ptype_abstract, Some { ptyp_desc = Ptyp_variant (constrs, Closed, None); ptyp_loc } ->
`Polymorphic,
List.fold_left (fun (acc, mappings) row_field ->
let error_inherit loc =
raise_errorf ~loc:ptyp_loc
"%s cannot be derived for inherited variant cases"
deriver
in
let error_arguments loc =
raise_errorf ~loc:ptyp_loc
"%s can be derived only for argumentless constructors"
deriver
in
let loc = row_field.prf_loc in
match row_field.prf_desc with
| Rinherit _ -> error_inherit loc
| Rtag (name, true, []) ->
map acc mappings rtag_attr_value row_field name
| Rtag _ -> error_arguments loc
)
(0, []) constrs
| _ -> raise_errorf ~loc:type_decl.ptype_loc
"%s can be derived only for variants" deriver
in
let rec check_dup mappings =
match mappings with
| (a, { txt=atxt; loc=aloc }) :: (b, { txt=btxt; loc=bloc }) :: _ when a = b ->
let sigil = match kind with `Regular -> "" | `Polymorphic -> "`" in
let sub =
[Ocaml_common.Location.errorf
~loc:bloc "Same as for %s%s" sigil btxt] in
raise_errorf ~sub ~loc:aloc
"%s: duplicate value %d for constructor %s%s" deriver a sigil atxt
| _ :: rest -> check_dup rest
| [] -> ()
in
mappings |> List.stable_sort (fun (a,_) (b,_) -> compare a b) |> check_dup;
kind, mappings
let str_of_type ({ ptype_loc = loc } as type_decl) =
let kind, mappings = mappings_of_type type_decl in
let patt name =
match kind with
| `Regular -> Pat.construct (mknoloc (Lident name)) None
| `Polymorphic -> Pat.variant name None
and expr name =
match kind with
| `Regular -> Exp.construct (mknoloc (Lident name)) None
| `Polymorphic -> Exp.variant name None
in
let to_enum_cases =
List.map (fun (value, { txt = name }) ->
Exp.case (patt name) (int value)) mappings
and from_enum_cases =
List.map (fun (value, { txt = name }) ->
Exp.case (pint value) (constr "Some" [expr name])) mappings @
[Exp.case (Pat.any ()) (constr "None" [])]
and indexes = List.map fst mappings in
[Vb.mk (pvar (Ppx_deriving.mangle_type_decl (`Prefix "min") type_decl))
(int (List.fold_left min max_int indexes));
Vb.mk (pvar (Ppx_deriving.mangle_type_decl (`Prefix "max") type_decl))
(int (List.fold_left max min_int indexes));
Vb.mk (pvar (Ppx_deriving.mangle_type_decl (`Suffix "to_enum") type_decl))
(Exp.function_ to_enum_cases);
Vb.mk (pvar (Ppx_deriving.mangle_type_decl (`Suffix "of_enum") type_decl))
(Exp.function_ from_enum_cases)]
let sig_of_type type_decl =
let loc = type_decl.ptype_loc in
let typ = Ppx_deriving.core_type_of_type_decl type_decl in
[Sig.value (Val.mk (mknoloc (Ppx_deriving.mangle_type_decl (`Prefix "min") type_decl))
[%type: Ppx_deriving_runtime.int]);
Sig.value (Val.mk (mknoloc (Ppx_deriving.mangle_type_decl (`Prefix "max") type_decl))
[%type: Ppx_deriving_runtime.int]);
Sig.value (Val.mk (mknoloc (Ppx_deriving.mangle_type_decl (`Suffix "to_enum") type_decl))
[%type: [%t typ] -> Ppx_deriving_runtime.int]);
Sig.value (Val.mk (mknoloc (Ppx_deriving.mangle_type_decl (`Suffix "of_enum") type_decl))
[%type: Ppx_deriving_runtime.int -> [%t typ] Ppx_deriving_runtime.option])]
let impl_generator = Deriving.Generator.V2.make_noarg (fun ~ctxt:_ (_, type_decls) ->
[Str.value Nonrecursive (List.concat (List.map str_of_type type_decls))])
let intf_generator = Deriving.Generator.V2.make_noarg (fun ~ctxt:_ (_, type_decls) ->
List.concat (List.map sig_of_type type_decls))
let deriving: Deriving.t =
Deriving.add
deriver
~str_type_decl:impl_generator
~sig_type_decl:intf_generator