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
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 =
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 value -> Format.fprintf fmt "\t%s = %s\n" key value)
(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 (
Lwt_log.ign_warning ~section "Number of connections now ok";
maxr := -1000;
Lwt_mvar.put mvar ())
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 ())