Source file ocsigen_cohttp.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
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
open Lwt.Infix
let section = Lwt_log.Section.make "ocsigen:cohttp"
exception Ocsigen_http_error of
Ocsigen_cookie_map.t * Cohttp.Code.status
exception Ext_http_error of
Cohttp.Code.status * string option * Cohttp.Header.t option
(** print_request Print request for debug
@param out_ch output for debug
@param request Cohttp request *)
let _print_request fmt request =
let print_list print_data out_ch lst =
let rec aux = function
| [] -> ()
| [ x ] -> print_data out_ch x
| x :: r -> print_data out_ch x; aux r
in aux lst
in
Format.fprintf fmt "%s [%s/%s]:\n"
(Uri.to_string (Cohttp.Request.uri request))
Cohttp.(Code.string_of_version (Request.version request))
Cohttp.(Code.string_of_method (Request.meth request));
Cohttp.Header.iter
(fun key values ->
(print_list
(fun fmt value -> Format.fprintf fmt "\t%s = %s\n" key value)
fmt
values))
(Cohttp.Request.headers request)
let connections = Hashtbl.create 256
let get_number_of_connected,
incr_connected,
decr_connected,
_wait_fewer_connected =
let connected = ref 0 in
let maxr = ref (-1000) in
let mvar = Lwt_mvar.create_empty () in
((fun () -> !connected),
(fun n -> connected := !connected + n),
(fun () ->
let c = !connected in
connected := c - 1;
if !connected < 0
then exit 1;
if c = !maxr
then begin
Lwt_log.ign_warning ~section "Number of connections now ok";
maxr := -1000;
Lwt_mvar.put mvar ()
end
else Lwt.return ()
),
(fun max ->
maxr := max;
Lwt_mvar.take mvar)
)
exception Ocsigen_is_dir of (Ocsigen_request.t -> Uri.t)
module Cookie = struct
let serialize_cookie_raw path exp name c secure =
Format.sprintf "%s=%s; path=/%s%s%s"
name c
(Ocsigen_lib.Url.string_of_url_path ~encode:true path)
(if secure then "; secure" else "")
(match exp with
| Some s ->
"; expires=" ^ (Ocsigen_lib.Date.to_string s)
| None ->
"")
let serialize_cookies path =
Ocsigen_cookie_map.Map_inner.fold @@ fun name c h ->
let open Ocsigen_cookie_map in
let exp, v, secure = match c with
| OUnset -> (Some 0., "", false)
| OSet (t, v, secure) -> (t, v, secure)
in
Cohttp.Header.add h
Ocsigen_header.Name.(to_string set_cookie)
(serialize_cookie_raw path exp name v secure)
let serialize cookies =
Ocsigen_cookie_map.Map_path.fold serialize_cookies cookies headers
end
let path exp name c _secure =
Format.sprintf "%s=%s%s%s" name c
("; path=/" ^ Ocsigen_lib.Url.string_of_url_path ~encode:true path)
"" ^
(match exp with
| Some s ->
"; expires=" ^ Ocsigen_lib.Date.to_string s
| None -> "")
let path t hds =
Ocsigen_cookie_map.Map_inner.fold
(fun name c h ->
let open Ocsigen_cookie_map in
let exp, v, secure =
match c with
| OUnset ->
Some 0., "", false
| OSet (t, v, secure) ->
t, v, secure
in
Cohttp.Header.add h
Ocsigen_header.Name.(to_string set_cookie)
(make_cookies_header path exp name v secure)
)
t
hds
let handler ~ssl ~address ~port ~connector (flow, conn) request body =
let filenames = ref [] in
let edn = Conduit_lwt_unix.endp_of_flow flow in
let rec getsockname = function
| `TCP (ip, port) ->
Unix.ADDR_INET (Ipaddr_unix.to_inet_addr ip, port)
| `Unix_domain_socket path ->
Unix.ADDR_UNIX path
| `TLS (_, edn) ->
getsockname edn
| `Unknown err ->
raise (Failure ("resolution failed: " ^ err))
| `Vchan_direct _ ->
raise (Failure "VChan not supported")
| `Vchan_domain_socket _ ->
raise (Failure "VChan not supported")
in
let sockaddr = getsockname edn in
let connection_closed =
try
fst (Hashtbl.find connections conn)
with Not_found ->
let ((connection_closed, _) as p) = Lwt.wait () in
Hashtbl.add connections conn p;
incr_connected 1;
connection_closed
in
let handle_error exn =
Lwt_log.ign_debug ~section ~exn "Got exception while handling request." ;
let , ret_code = match exn with
| Ocsigen_http_error (cookies_to_set, code) ->
let =
Cookie.serialize cookies_to_set (Cohttp.Header.init ())
in
Some headers, code
| Ocsigen_stream.Interrupted Ocsigen_stream.Already_read ->
None, `Internal_server_error
| Unix.Unix_error (Unix.EACCES, _, _) ->
None, `Forbidden
| Ext_http_error (code, _, ) ->
headers, code
| Ocsigen_lib.Ocsigen_Bad_Request ->
None, `Bad_request
| Ocsigen_lib.Ocsigen_Request_too_long ->
None, `Request_entity_too_large
| exn ->
Lwt_log.ign_error ~section ~exn "Error while handling request." ;
None, `Internal_server_error
in
let body =
match ret_code with
| `Not_found -> "Not Found"
| _ -> Printexc.to_string exn in
Cohttp_lwt_unix.Server.respond_error
?headers ~status:(ret_code :> Cohttp.Code.status_code) ~body ()
in
let request =
Ocsigen_request.make
~address ~port ~ssl
~filenames ~sockaddr ~body ~connection_closed request
in
Lwt.finalize
(fun () ->
Ocsigen_messages.accesslog
(Format.sprintf
"connection for %s from %s (%s)%s: %s"
(match Ocsigen_request.host request with
| None -> "<host not specified in the request>"
| Some h -> h)
(Ocsigen_request.remote_ip request)
(Option.value ~default:"" (Ocsigen_request.header request
Ocsigen_header.Name.user_agent))
(Option.fold ~none:"" ~some:(fun s -> " X-Forwarded-For: " ^ s)
(Ocsigen_request.header request
Ocsigen_header.Name.x_forwarded_for))
(Uri.path (Ocsigen_request.uri request))
);
Lwt.catch
(fun () ->
connector request >>= fun response ->
let response, body = Ocsigen_response.to_cohttp response
and cookies = Ocsigen_response.cookies response in
let response =
let =
Cohttp.Header.add_unless_exists
(Cohttp.Header.add_unless_exists
(Ocsigen_cookie_map.Map_path.fold
make_cookies_headers
cookies
(Cohttp.Response.headers response))
"server" Ocsigen_config.server_name)
"date"
(Ocsigen_lib.Date.to_string
(Unix.time ()))
in
{ response with Cohttp.Response.headers }
in
Lwt.return (response, body))
(function
| Ocsigen_is_dir fun_request ->
let =
fun_request request
|> Uri.to_string
|> Cohttp.Header.init_with "location"
and status = `Moved_permanently in
Cohttp_lwt_unix.Server.respond ~headers ~status ~body:`Empty ()
| exn ->
handle_error exn))
(fun () ->
if !filenames <> [] then
List.iter
(fun a ->
try
Unix.unlink a
with Unix.Unix_error _ as exn ->
Lwt_log.ign_warning_f ~section ~exn
"Error while removing file %s" a)
!filenames;
Lwt.return_unit)
let conn_closed (_flow, conn) =
try
Lwt_log.ign_debug_f ~section
"Connection closed:\n%s"
(Cohttp.Connection.to_string conn);
Lwt.wakeup (snd (Hashtbl.find connections conn)) ();
Hashtbl.remove connections conn;
Lwt.async decr_connected
with Not_found -> ()
let stop, stop_wakener = Lwt.wait ()
let shutdown timeout =
let process =
match timeout with
| Some f -> (fun () -> Lwt_unix.sleep f)
| None -> (fun () -> Lwt.return ())
in
ignore (Lwt.pick [process (); stop] >>= fun () -> exit 0)
let service ?ssl ~address ~port ~connector () =
let tls_own_key =
match ssl with
| Some (crt, key, Some password) ->
`TLS (`Crt_file_path crt,
`Key_file_path key,
`Password password)
| Some (crt, key, None) ->
`TLS (`Crt_file_path crt,
`Key_file_path key,
`No_password)
| None -> `None
in
Conduit_lwt_unix.init
~src:(Ocsigen_config.Socket_type.to_string address)
~tls_own_key () >>= fun conduit_ctx ->
Lwt.return (Cohttp_lwt_unix.Net.init ~ctx:conduit_ctx ()) >>= fun ctx ->
let callback =
let address = Ocsigen_config.Socket_type.to_inet_addr address
and ssl = match ssl with Some _ -> true | None -> false in
handler ~ssl ~address ~port ~connector
in
let config = Cohttp_lwt_unix.Server.make ~conn_closed ~callback () in
let mode =
match tls_own_key with
| `None -> `TCP (`Port port)
| `TLS (crt, key, pass) ->
`OpenSSL (crt, key, pass, `Port port)
in
Cohttp_lwt_unix.Server.create ~stop ~ctx ~mode config
>>= fun () ->
Lwt.return (Lwt.wakeup stop_wakener ())