-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathconfiguration.ml
137 lines (130 loc) · 4.54 KB
/
configuration.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
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
open Utils.Json
let current_version = 1
type t = {
certificate : X509.Certificate.t;
private_key : X509.Private_key.t;
server_ip : Ipaddr.t;
server_port : int;
updated_at : Ptime.t;
}
(* this is here to avoid the need for option *)
let empty () =
let key = X509.Private_key.generate `ED25519 in
let name = "mollymawk-empty" in
let cert =
let dn =
X509.Distinguished_name.
[ Relative_distinguished_name.(singleton (CN name)) ]
in
let csr = Result.get_ok (X509.Signing_request.create dn key) in
let valid_from = Ptime.epoch in
let valid_until =
Option.get (Ptime.add_span valid_from (Ptime.Span.of_int_s (10 * 60)))
in
Result.get_ok
(X509.Signing_request.sign csr ~valid_from ~valid_until key dn)
in
{
certificate = cert;
private_key = key;
server_ip = Ipaddr.(V4 V4.any);
server_port = 1025;
updated_at = Ptime.epoch;
}
let to_json t =
`Assoc
[
("version", `Int current_version);
("certificate", `String (X509.Certificate.encode_pem t.certificate));
("private_key", `String (X509.Private_key.encode_pem t.private_key));
("server_ip", `String (Ipaddr.to_string t.server_ip));
("server_port", `Int t.server_port);
("updated_at", `String (Utils.TimeHelper.string_of_ptime t.updated_at));
]
let of_json_from_http json_dict now =
match
( get "certificate" json_dict,
get "private_key" json_dict,
get "server_ip" json_dict,
get "server_port" json_dict )
with
| ( Some (`String cert),
Some (`String key),
Some (`String server_ip),
Some (`Int server_port) ) ->
let ( let* ) = Result.bind in
let* certificate = X509.Certificate.decode_pem cert in
let* private_key = X509.Private_key.decode_pem key in
let* () =
if
not
(String.equal
(X509.Public_key.fingerprint
(X509.Certificate.public_key certificate))
(X509.Public_key.fingerprint
(X509.Private_key.public private_key)))
then Error (`Msg "certificate and private key do not match")
else Ok ()
in
let* server_ip = Ipaddr.of_string server_ip in
Ok { certificate; private_key; server_ip; server_port; updated_at = now }
| _ ->
Error
(`Msg
(Fmt.str "configuration: unexpected types, got %s"
(Yojson.Basic.to_string (`Assoc json_dict))))
let of_json json =
match json with
| `Assoc xs -> (
match get "version" xs with
| None -> Error (`Msg "configuration: couldn't find a version")
| Some (`Int v) ->
if v = current_version then
match
( get "certificate" xs,
get "private_key" xs,
get "server_ip" xs,
get "server_port" xs,
get "updated_at" xs )
with
| ( Some (`String cert),
Some (`String key),
Some (`String server_ip),
Some (`Int server_port),
Some (`String updated_at) ) ->
let ( let* ) = Result.bind in
let* certificate = X509.Certificate.decode_pem cert in
let* private_key = X509.Private_key.decode_pem key in
let* () =
if
not
(String.equal
(X509.Public_key.fingerprint
(X509.Certificate.public_key certificate))
(X509.Public_key.fingerprint
(X509.Private_key.public private_key)))
then Error (`Msg "certificate and private key do not match")
else Ok ()
in
let* server_ip = Ipaddr.of_string server_ip in
let* updated_at = Utils.TimeHelper.ptime_of_string updated_at in
Ok
{
certificate;
private_key;
server_ip;
server_port;
updated_at;
}
| _ ->
Error
(`Msg
(Fmt.str "configuration: unexpected types, got %s"
(Yojson.Basic.to_string (`Assoc xs))))
else
Error
(`Msg
(Fmt.str "configuration: found version %u, expected %u" v
current_version))
| Some _ -> Error (`Msg "configuration: version must be an integer"))
| _ -> Error (`Msg "configuration: expected a dictionary")