-
Notifications
You must be signed in to change notification settings - Fork 177
/
Copy pathtest_request.ml
409 lines (349 loc) · 14 KB
/
test_request.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
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
open Cohttp
module String_io = Cohttp.Private.String_io
module StringRequest = Request.Private.Make (String_io.M)
let user_agent = Cohttp.Header.user_agent
let uri_userinfo = Uri.of_string "http://foo:bar%2525@ocaml.org"
let header_auth =
let h = Header.init () in
let h = Header.add_authorization h (`Basic ("qux", "qwerty")) in
h
let is_some = function None -> false | Some _ -> true
let header_has_auth _ =
Alcotest.check Alcotest.bool "Test header has auth"
(header_auth |> Header.get_authorization |> is_some)
true
let uri_has_userinfo _ =
Alcotest.check Alcotest.bool "Uri has user info"
(uri_userinfo |> Uri.userinfo |> is_some)
true
let t_credentials =
Alcotest.testable
(fun fmt c ->
let sexp = Cohttp.Auth.sexp_of_credential c in
Sexplib0.Sexp.pp_hum fmt sexp)
( = )
let auth_uri_no_override _ =
let r = Request.make ~headers:header_auth uri_userinfo in
Alcotest.check
(Alcotest.option t_credentials)
"auth uri no override"
(r |> Request.headers |> Header.get_authorization)
(Header.get_authorization header_auth)
let auth_uri _ =
let r = Request.make uri_userinfo in
Alcotest.check
(Alcotest.option t_credentials)
"auth_uri"
(r |> Request.headers |> Header.get_authorization)
(Some (`Basic ("foo", "bar%25")))
let t_encoding =
Alcotest.testable
(fun fmt e ->
let sexp = Cohttp.Transfer.sexp_of_encoding e in
Sexplib0.Sexp.pp fmt sexp)
( = )
let encoding_content_length_header () =
let r =
Request.make
~headers:(Cohttp.Header.of_list [ ("content-length", "100") ])
(Uri.of_string "http://someuri.com")
in
Alcotest.check t_encoding "body encoding determined by content-length header"
(r |> Request.encoding) (Fixed 100L)
let encoding_transfer_encoding_header () =
let r =
Request.make
~headers:(Cohttp.Header.of_list [ ("transfer-encoding", "chunked") ])
(Uri.of_string "http://someuri.com")
in
Alcotest.check t_encoding
"body encoding determined by transfer-encoding header"
(r |> Request.encoding) Chunked
let encoding_both_headers () =
let r =
Request.make
~headers:
(Cohttp.Header.of_list
[ ("transfer-encoding", "chunked"); ("content-length", "100") ])
(Uri.of_string "http://someuri.com")
in
Alcotest.check t_encoding
"body encoding with content-length and transfer-encoding headers."
(r |> Request.encoding) Chunked
let encoding_header_opt_argument () =
let r =
Request.make ~encoding:Chunked
~headers:(Cohttp.Header.of_list [ ("content-length", "100") ])
(Uri.of_string "http://someuri.com")
in
Alcotest.check t_encoding
"body encoding with content-length and transfer-encoding headers."
(r |> Request.encoding) (Fixed 100L)
module Parse_result = struct
type 'a t = [ `Ok of 'a | `Invalid of string | `Eof ]
let map t ~f =
match t with `Ok x -> `Ok (f x) | (`Invalid _ | `Eof) as e -> e
end
let uri_testable : Uri.t Alcotest.testable =
Alcotest.testable Uri.pp_hum Uri.equal
let t_parse_result_uri : Uri.t Parse_result.t Alcotest.testable =
Alcotest.testable
(fun fmt -> function
| `Invalid s -> Format.fprintf fmt "`Invalid %s" s
| `Eof -> Format.fprintf fmt "`Eof"
| `Ok u -> Uri.pp_hum fmt u)
(fun x y ->
match (x, y) with `Ok x, `Ok y -> Uri.equal x y | x, y -> x = y)
let parse_request_uri_ r (expected : Uri.t Parse_result.t) name =
String_io.M.(
StringRequest.read (String_io.open_in r)
>>= fun (result : Http.Request.t Parse_result.t) ->
let uri = Parse_result.map result ~f:Request.uri in
return @@ Alcotest.check t_parse_result_uri name uri expected)
let bad_request = `Invalid "bad request URI"
let parse_request_uri _ =
let r = "GET / HTTP/1.1\r\n\r\n" in
let uri = `Ok (Uri.of_string "/") in
parse_request_uri_ r uri "parse_request_uri"
let parse_request_uri_host _ =
let r = "GET / HTTP/1.1\r\nHost: example.com\r\n\r\n" in
let uri = `Ok (Uri.of_string "//example.com/") in
parse_request_uri_ r uri "parse_request_uri_host"
let parse_request_uri_host_port _ =
let r = "GET / HTTP/1.1\r\nHost: example.com:8080\r\n\r\n" in
let uri = `Ok (Uri.of_string "//example.com:8080/") in
parse_request_uri_ r uri "parse_request_uri_host_port"
let parse_request_uri_double_slash _ =
let r = "GET // HTTP/1.1\r\n\r\n" in
let uri = `Ok (Uri.with_path (Uri.of_string "") "//") in
parse_request_uri_ r uri "parse_request_uri_double_slash"
let parse_request_uri_host_double_slash _ =
let r = "GET // HTTP/1.1\r\nHost: example.com\r\n\r\n" in
let uri = `Ok (Uri.of_string "//example.com//") in
parse_request_uri_ r uri "parse_request_uri_host_double_slash"
let parse_request_uri_triple_slash _ =
let r = "GET /// HTTP/1.1\r\n\r\n" in
let uri = `Ok (Uri.with_path (Uri.of_string "") "///") in
parse_request_uri_ r uri "parse_request_uri_triple_slash"
let parse_request_uri_host_triple_slash _ =
let r = "GET /// HTTP/1.1\r\nHost: example.com\r\n\r\n" in
let uri = `Ok (Uri.of_string "//example.com///") in
parse_request_uri_ r uri "parse_request_uri_host_triple_slash"
let parse_request_uri_no_slash _ =
let r = "GET foo HTTP/1.1\r\n\r\n" in
parse_request_uri_ r bad_request "parse_request_uri_no_slash"
let parse_request_uri_host_no_slash _ =
let r = "GET foo HTTP/1.1\r\nHost: example.com\r\n\r\n" in
parse_request_uri_ r bad_request "parse_request_uri_host_no_slash"
let parse_request_uri_empty _ =
let r = "GET HTTP/1.1\r\n\r\n" in
let uri = `Ok (Uri.of_string "/") in
parse_request_uri_ r uri "parse_request_uri_empty"
let parse_request_uri_host_empty _ =
let r = "GET HTTP/1.1\r\nHost: example.com\r\n\r\n" in
let uri = `Ok (Uri.of_string "//example.com/") in
parse_request_uri_ r uri "parse_request_uri_host_empty"
let parse_request_uri_path_like_scheme _ =
let r = "GET http://example.net HTTP/1.1\r\n\r\n" in
let uri = `Ok (Uri.of_string "http://example.net/") in
parse_request_uri_ r uri "parse_request_uri_path_like_scheme"
let parse_request_uri_host_path_like_scheme _ =
let r = "GET http://example.net HTTP/1.1\r\nHost: example.com\r\n\r\n" in
let uri = `Ok (Uri.of_string "http://example.net/") in
parse_request_uri_ r uri "parse_request_uri_host_path_like_scheme"
let parse_request_uri_path_like_host_port _ =
let path = "//example.net:8080" in
let r = "GET " ^ path ^ " HTTP/1.1\r\n\r\n" in
let uri = `Ok (Uri.with_path (Uri.of_string "") path) in
parse_request_uri_ r uri "parse_request_uri_path_like_host_port"
let parse_request_uri_host_path_like_host_port _ =
let path = "//example.net:8080" in
let r = "GET " ^ path ^ " HTTP/1.1\r\nHost: example.com\r\n\r\n" in
let uri = `Ok (Uri.with_path (Uri.of_string "//example.com") path) in
parse_request_uri_ r uri "parse_request_uri_host_path_like_host_port"
let parse_request_uri_query _ =
let pqs = "/?foo" in
let r = "GET " ^ pqs ^ " HTTP/1.1\r\n\r\n" in
let uri = `Ok (Uri.of_string pqs) in
parse_request_uri_ r uri "parse_request_uri_query"
let parse_request_uri_host_query _ =
let pqs = "/?foo" in
let r = "GET " ^ pqs ^ " HTTP/1.1\r\nHost: example.com\r\n\r\n" in
let uri = `Ok (Uri.of_string ("//example.com" ^ pqs)) in
parse_request_uri_ r uri "parse_request_uri_host_query"
let parse_request_uri_query_no_slash _ =
let r = "GET ?foo HTTP/1.1\r\n\r\n" in
parse_request_uri_ r bad_request "parse_request_uri_query_no_slash"
let parse_request_uri_host_query_no_slash _ =
let r = "GET ?foo HTTP/1.1\r\nHost: example.com\r\n\r\n" in
parse_request_uri_ r bad_request "parse_request_uri_host_query_no_slash"
let parse_request_connect _ =
let r = "CONNECT vpn.example.net:443 HTTP/1.1\r\n\r\n" in
let uri = `Ok (Uri.of_string "//vpn.example.net:443") in
parse_request_uri_ r uri "parse_request_connect"
let parse_request_connect_host _ =
let r =
"CONNECT vpn.example.net:443 HTTP/1.1\r\nHost: vpn.example.com:443\r\n\r\n"
in
let uri = `Ok (Uri.of_string "//vpn.example.net:443") in
parse_request_uri_ r uri "parse_request_connect_host"
let parse_request_options _ =
let r = "OPTIONS * HTTP/1.1\r\n\r\n" in
let uri = `Ok (Uri.of_string "") in
parse_request_uri_ r uri "parse_request_options"
let parse_request_options_host _ =
let r = "OPTIONS * HTTP/1.1\r\nHost: example.com:443\r\n\r\n" in
let uri = `Ok (Uri.of_string "//example.com:443") in
parse_request_uri_ r uri "parse_request_options_host"
let parse_request_uri_traversal _ =
let r = "GET /../../../../etc/shadow HTTP/1.1\r\n\r\n" in
let uri = `Ok (Uri.of_string "/etc/shadow") in
parse_request_uri_ r uri "parse_request_uri_traversal"
let parse_request_uri_host_traversal _ =
let r = "GET /../../../../etc/shadow HTTP/1.1\r\nHost: example.com\r\n\r\n" in
let uri = `Ok (Uri.of_string "//example.com/etc/shadow") in
parse_request_uri_ r uri "parse_request_uri_host_traversal"
let uri_round_trip _ =
let expected_uri = Uri.of_string "https://www.example.com/test" in
let actual_uri = Request.make expected_uri |> Request.uri in
Alcotest.check uri_testable "Request.make uri round-trip" actual_uri
expected_uri
let () = Printexc.record_backtrace true
module Buffer = struct
include Http_bytebuffer.Bytebuffer
include
Http_bytebuffer.Bytebuffer.Make
(struct
type 'a t = 'a
let ( >>= ) v f = f v
let ( >>| ) v f = f v
let return v = v
end)
(struct
type src = string
let refill s buffer ~pos ~len =
if String.equal s "" then `Eof
else
let len = min len (String.length s) in
let () = Bytes.blit_string s 0 buffer pos len in
`Ok len
end)
end
module Test_io = struct
type 'a t = 'a
let ( >>= ) v f = f v
let return v = v
type ic = Buffer.t
type oc = Buffer.t
type conn = unit
let refill _ = `Eof
let with_input_buffer b ~f =
let contents = Buffer.to_string b in
let res, read = f contents ~pos:0 ~len:(String.length contents) in
let () = Buffer.drop b read in
res
let read_line buffer = Buffer.read_line buffer ""
let read buffer = Buffer.read buffer ""
let write buffer string = Buffer.refill buffer string |> ignore
let flush _ = ()
end
module Request = Request.Private.Make (Test_io)
let null_content_length_header () =
let output = Buffer.create 1024 in
let () =
(* The user-agent in releases contentsontains the version, we need to strip
it for the test *)
let r =
Cohttp.Request.make_for_client ~chunked:false ~body_length:0L `PUT
(Uri.of_string "http://someuri.com")
in
Request.write_header r output
in
let expected =
"PUT / HTTP/1.1\r\nhost: someuri.com\r\nuser-agent: "
^ user_agent
^ "\r\ncontent-length: 0\r\n\r\n"
in
Alcotest.(check string)
"null content-length header are sent" expected (Buffer.to_string output)
let useless_null_content_length_header () =
let output = Buffer.create 1024 in
let () =
let r =
Cohttp.Request.make_for_client `GET (Uri.of_string "http://someuri.com")
in
Request.write_header r output
in
let expected =
"GET / HTTP/1.1\r\nhost: someuri.com\r\nuser-agent: "
^ user_agent
^ "\r\n\r\n"
in
Alcotest.(check string)
"null content-length header are not sent for bodyless methods" expected
(Buffer.to_string output)
let () =
Alcotest.run "test_request"
[
( "Auth",
[
("header has auth", `Quick, header_has_auth);
("URI has user info", `Quick, uri_has_userinfo);
("from URI - do not override", `Quick, auth_uri_no_override);
("from URI", `Quick, auth_uri);
] );
( "Encoding",
[
("from content-length header", `Quick, encoding_content_length_header);
( "from transfer-encoding header",
`Quick,
encoding_transfer_encoding_header );
("with both headers", `Quick, encoding_both_headers);
( "from both optional argument and headers",
`Quick,
encoding_header_opt_argument );
("null content-length", `Quick, null_content_length_header);
( "useless null content-length",
`Quick,
useless_null_content_length_header );
] );
( "Parse URI",
[
("simple", `Quick, parse_request_uri);
("with host", `Quick, parse_request_uri_host);
("with host and port", `Quick, parse_request_uri_host_port);
("double slash", `Quick, parse_request_uri_double_slash);
("double slash with host", `Quick, parse_request_uri_host_double_slash);
("triple slash", `Quick, parse_request_uri_triple_slash);
("triple slash with host", `Quick, parse_request_uri_host_triple_slash);
("no slash", `Quick, parse_request_uri_no_slash);
("no slash with host", `Quick, parse_request_uri_host_no_slash);
("empty", `Quick, parse_request_uri_empty);
("empty with host", `Quick, parse_request_uri_host_empty);
("path like scheme", `Quick, parse_request_uri_path_like_scheme);
( "path like scheme with host",
`Quick,
parse_request_uri_host_path_like_scheme );
("path like host:port", `Quick, parse_request_uri_path_like_host_port);
( "path like host:port with host",
`Quick,
parse_request_uri_host_path_like_host_port );
("with query string", `Quick, parse_request_uri_query);
("with query with host", `Quick, parse_request_uri_host_query);
( "no slash with query string",
`Quick,
parse_request_uri_query_no_slash );
( "no slash with query with host",
`Quick,
parse_request_uri_host_query_no_slash );
("CONNECT", `Quick, parse_request_connect);
("CONNECT with host", `Quick, parse_request_connect_host);
("OPTIONS", `Quick, parse_request_options);
("OPTIONS with host", `Quick, parse_request_options_host);
("parent traversal", `Quick, parse_request_uri_traversal);
( "parent traversal with host",
`Quick,
parse_request_uri_host_traversal );
("uri round-trip", `Quick, uri_round_trip);
] );
]