-
Notifications
You must be signed in to change notification settings - Fork 3
/
storage.ml
226 lines (201 loc) · 7.08 KB
/
storage.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
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
open Utils.Json
let current_version = 6
(* version history:
1 was initial (fields until email_verification_uuid)
2 added active
3 added super_user (but did not serialise it)
4 properly serialised super_user
5 cookie has two new fields last_access and user_agent
6 tokens has 3 new fields: name, last_access and usage_count
*)
let t_to_json users configuration =
`Assoc
[
("version", `Int current_version);
("users", `List (List.map User_model.user_to_json users));
("configuration", Configuration.to_json configuration);
]
let t_of_json json =
match json with
| `Assoc xs -> (
let ( let* ) = Result.bind in
match (get "version" xs, get "users" xs, get "configuration" xs) with
| Some (`Int v), Some (`List users), Some configuration ->
let* () =
if v = current_version then Ok ()
else if v = 5 then Ok ()
else if v = 4 then Ok ()
else if v = 3 then Ok ()
else if v = 2 then Ok ()
else if v = 1 then Ok ()
else
Error
(`Msg
(Fmt.str "expected version %u, found version %u"
current_version v))
in
let* users =
List.fold_left
(fun acc js ->
let* acc = acc in
let* user =
if v = 1 then User_model.user_v1_of_json js
else if v = 2 || v = 3 then User_model.user_v2_of_json js
else if v = 4 || v = 5 then
User_model.(user_v3_of_json cookie_v1_of_json) js
else User_model.(user_of_json cookie_of_json) js
in
Ok (user :: acc))
(Ok []) users
in
let* configuration = Configuration.of_json configuration in
Ok (users, configuration)
| _ -> Error (`Msg "invalid data: no version and users field"))
| _ -> Error (`Msg "invalid data: not an assoc")
let error_msgf fmt = Fmt.kstr (fun msg -> Error (`Msg msg)) fmt
module Make (BLOCK : Mirage_block.S) = struct
module Stored_data = OneFFS.Make (BLOCK)
open Lwt.Infix
type t = {
disk : Stored_data.t;
mutable users : User_model.user list;
mutable configuration : Configuration.t;
}
let write_data t =
Stored_data.write t.disk
(Yojson.Basic.to_string (t_to_json t.users t.configuration))
let read_data disk =
Stored_data.read disk >|= function
| Ok (Some s) ->
let ( let* ) = Result.bind in
let* json =
try Ok (Yojson.Basic.from_string s)
with Yojson.Json_error msg -> Error (`Msg ("Invalid json: " ^ msg))
in
let* t = t_of_json json in
Ok t
| Ok None -> Ok ([], Configuration.empty ())
| Error e ->
error_msgf "error while reading storage: %a" Stored_data.pp_error e
let connect block =
Stored_data.connect block >>= fun disk ->
read_data disk >|= function
| Error _ as e -> e
| Ok (users, configuration) -> Ok { disk; users; configuration }
let configuration { configuration; _ } = configuration
let update_configuration t (configuration : Configuration.t) =
let t' = { t with configuration } in
write_data t' >|= function
| Ok () ->
t.configuration <- configuration;
Ok ()
| Error we ->
error_msgf "error while writing storage: %a" Stored_data.pp_write_error
we
let add_user t user =
let t' = { t with users = user :: t.users } in
write_data t' >|= function
| Ok () ->
t.users <- user :: t.users;
Ok ()
| Error we ->
error_msgf "error while writing storage: %a" Stored_data.pp_write_error
we
let update_user t (user : User_model.user) =
let users =
List.map
(fun (u : User_model.user) ->
match u.uuid = user.uuid with true -> user | false -> u)
t.users
in
let t' = { t with users } in
write_data t' >|= function
| Ok () ->
t.users <- users;
Ok ()
| Error we ->
error_msgf "error while writing storage: %a" Stored_data.pp_write_error
we
let users { users; _ } = users
let find_by_email store email =
List.find_opt
(fun user -> String.equal user.User_model.email email)
store.users
let find_by_name store name =
List.find_opt
(fun user -> String.equal user.User_model.name name)
store.users
let find_by_uuid store uuid =
List.find_opt
(fun user -> String.equal user.User_model.uuid uuid)
store.users
let find_by_cookie store cookie_value =
List.fold_left
(fun acc user ->
match acc with
| Some _ as s -> s
| None -> (
match
List.find_opt
(fun (cookie : User_model.cookie) ->
String.equal User_model.session_cookie cookie.User_model.name
&& String.equal cookie_value cookie.value)
user.User_model.cookies
with
| None -> None
| Some c -> Some (user, c)))
None store.users
let find_by_api_token store token =
List.find_map
(fun (user : User_model.user) ->
match
List.find_opt
(fun (token_ : User_model.token) -> String.equal token token_.value)
user.tokens
with
| Some token_ -> Some (user, token_)
| None -> None)
store.users
let increment_token_usage store (token : User_model.token)
(user : User_model.user) =
let token = { token with usage_count = token.usage_count + 1 } in
let tokens =
List.map
(fun (token' : User_model.token) ->
if String.equal token.value token'.value then token else token')
user.tokens
in
let updated_user = User_model.update_user user ~tokens () in
update_user store updated_user >>= function
| Ok () -> Lwt.return (Ok ())
| Error (`Msg err) ->
Logs.err (fun m -> m "Error with storage: %s" err);
Lwt.return (Error (`Msg err))
let update_cookie_usage store (cookie : User_model.cookie)
(user : User_model.user) reqd =
let cookie = { cookie with user_agent = Middleware.user_agent reqd } in
let cookies =
List.map
(fun (cookie' : User_model.cookie) ->
if String.equal cookie.value cookie'.value then cookie else cookie')
user.cookies
in
let updated_user = User_model.update_user user ~cookies () in
update_user store updated_user >>= function
| Ok () -> Lwt.return (Ok ())
| Error (`Msg err) ->
Logs.err (fun m -> m "Error with storage: %s" err);
Lwt.return (Error (`Msg err))
let count_users store = List.length store.users
let find_email_verification_token store uuid =
List.find_opt
(fun user ->
Option.fold ~none:false
~some:(fun uu -> Uuidm.equal uu uuid)
user.User_model.email_verification_uuid)
store.users
let count_active store =
List.length (List.filter (fun u -> u.User_model.active) store.users)
let count_superusers store =
List.length (List.filter (fun u -> u.User_model.super_user) store.users)
end