mirror of
https://github.com/c-cube/tiny_httpd.git
synced 2025-12-06 03:05:29 -05:00
basic test for moonpool-io
This commit is contained in:
parent
c43ffb5ff4
commit
cf9c14b1c2
5 changed files with 331 additions and 0 deletions
|
|
@ -14,6 +14,12 @@
|
||||||
(modules echo vfs)
|
(modules echo vfs)
|
||||||
(libraries tiny_httpd logs tiny_httpd_camlzip))
|
(libraries tiny_httpd logs tiny_httpd_camlzip))
|
||||||
|
|
||||||
|
(executable
|
||||||
|
(name echo_mio)
|
||||||
|
(flags :standard -warn-error -a+8)
|
||||||
|
(modules echo_mio)
|
||||||
|
(libraries tiny_httpd tiny_httpd_moonpool logs))
|
||||||
|
|
||||||
(executable
|
(executable
|
||||||
(name writer)
|
(name writer)
|
||||||
(flags :standard -warn-error -a+8)
|
(flags :standard -warn-error -a+8)
|
||||||
|
|
|
||||||
282
examples/echo_mio.ml
Normal file
282
examples/echo_mio.ml
Normal file
|
|
@ -0,0 +1,282 @@
|
||||||
|
open Tiny_httpd_core
|
||||||
|
module Log = Tiny_httpd.Log
|
||||||
|
|
||||||
|
let now_ = Unix.gettimeofday
|
||||||
|
|
||||||
|
let alice_text =
|
||||||
|
"CHAPTER I. Down the Rabbit-Hole Alice was beginning to get very tired of \
|
||||||
|
sitting by her sister on the bank, and of having nothing to do: once or \
|
||||||
|
twice she had peeped into the book her sister was reading, but it had no \
|
||||||
|
pictures or conversations in it, <and what is the use of a book,> thought \
|
||||||
|
Alice <without pictures or conversations?> So she was considering in her \
|
||||||
|
own mind (as well as she could, for the hot day made her feel very sleepy \
|
||||||
|
and stupid), whether the pleasure of making a daisy-chain would be worth \
|
||||||
|
the trouble of getting up and picking the daisies, when suddenly a White \
|
||||||
|
Rabbit with pink eyes ran close by her. There was nothing so very \
|
||||||
|
remarkable in that; nor did Alice think it so very much out of the way to \
|
||||||
|
hear the Rabbit say to itself, <Oh dear! Oh dear! I shall be late!> (when \
|
||||||
|
she thought it over afterwards, it occurred to her that she ought to have \
|
||||||
|
wondered at this, but at the time it all seemed quite natural); but when \
|
||||||
|
the Rabbit actually took a watch out of its waistcoat-pocket, and looked at \
|
||||||
|
it, and then hurried on, Alice started to her feet, for it flashed across \
|
||||||
|
her mind that she had never before seen a rabbit with either a \
|
||||||
|
waistcoat-pocket, or a watch to take out of it, and burning with curiosity, \
|
||||||
|
she ran across the field after it, and fortunately was just in time to see \
|
||||||
|
it pop down a large rabbit-hole under the hedge. In another moment down \
|
||||||
|
went Alice after it, never once considering how in the world she was to get \
|
||||||
|
out again. The rabbit-hole went straight on like a tunnel for some way, and \
|
||||||
|
then dipped suddenly down, so suddenly that Alice had not a moment to think \
|
||||||
|
about stopping herself before she found herself falling down a very deep \
|
||||||
|
well. Either the well was very deep, or she fell very slowly, for she had \
|
||||||
|
plenty of time as she went down to look about her and to wonder what was \
|
||||||
|
going to happen next. First, she tried to look down and make out what she \
|
||||||
|
was coming to, but it was too dark to see anything; then she looked at the \
|
||||||
|
sides of the well, and noticed that they were filled with cupboards......"
|
||||||
|
|
||||||
|
(* util: a little middleware collecting statistics *)
|
||||||
|
let middleware_stat () : Server.Middleware.t * (unit -> string) =
|
||||||
|
let n_req = ref 0 in
|
||||||
|
let total_time_ = ref 0. in
|
||||||
|
let parse_time_ = ref 0. in
|
||||||
|
let build_time_ = ref 0. in
|
||||||
|
let write_time_ = ref 0. in
|
||||||
|
|
||||||
|
let m h req ~resp =
|
||||||
|
incr n_req;
|
||||||
|
let t1 = Request.start_time req in
|
||||||
|
let t2 = now_ () in
|
||||||
|
h req ~resp:(fun response ->
|
||||||
|
let t3 = now_ () in
|
||||||
|
resp response;
|
||||||
|
let t4 = now_ () in
|
||||||
|
total_time_ := !total_time_ +. (t4 -. t1);
|
||||||
|
parse_time_ := !parse_time_ +. (t2 -. t1);
|
||||||
|
build_time_ := !build_time_ +. (t3 -. t2);
|
||||||
|
write_time_ := !write_time_ +. (t4 -. t3))
|
||||||
|
and get_stat () =
|
||||||
|
Printf.sprintf
|
||||||
|
"%d requests (average response time: %.3fms = %.3fms + %.3fms + %.3fms)"
|
||||||
|
!n_req
|
||||||
|
(!total_time_ /. float !n_req *. 1e3)
|
||||||
|
(!parse_time_ /. float !n_req *. 1e3)
|
||||||
|
(!build_time_ /. float !n_req *. 1e3)
|
||||||
|
(!write_time_ /. float !n_req *. 1e3)
|
||||||
|
in
|
||||||
|
m, get_stat
|
||||||
|
|
||||||
|
(* ugly AF *)
|
||||||
|
let base64 x =
|
||||||
|
let ic, oc = Unix.open_process "base64" in
|
||||||
|
output_string oc x;
|
||||||
|
flush oc;
|
||||||
|
close_out oc;
|
||||||
|
let r = input_line ic in
|
||||||
|
ignore (Unix.close_process (ic, oc));
|
||||||
|
r
|
||||||
|
|
||||||
|
let setup_logging () =
|
||||||
|
Logs.set_reporter @@ Logs.format_reporter ();
|
||||||
|
Logs.set_level ~all:true (Some Logs.Debug)
|
||||||
|
|
||||||
|
let () =
|
||||||
|
Moonpool_fib.main @@ fun _ ->
|
||||||
|
let port_ = ref 8080 in
|
||||||
|
let max_conn = ref 800 in
|
||||||
|
let j = ref 16 in
|
||||||
|
Arg.parse
|
||||||
|
(Arg.align
|
||||||
|
[
|
||||||
|
"--port", Arg.Set_int port_, " set port";
|
||||||
|
"-p", Arg.Set_int port_, " set port";
|
||||||
|
"--debug", Arg.Unit setup_logging, " enable debug";
|
||||||
|
( "--max-connections",
|
||||||
|
Arg.Set_int max_conn,
|
||||||
|
" maximum number of connections" );
|
||||||
|
"-j", Arg.Set_int j, " Size of thread pool";
|
||||||
|
])
|
||||||
|
(fun _ -> raise (Arg.Bad ""))
|
||||||
|
"echo [option]*";
|
||||||
|
|
||||||
|
let runner = Moonpool.Ws_pool.create ~num_threads:!j () in
|
||||||
|
let server : Server.t =
|
||||||
|
Tiny_httpd_moonpool.create ~runner ~port:!port_ ~max_connections:!max_conn
|
||||||
|
()
|
||||||
|
in
|
||||||
|
|
||||||
|
let m_stats, get_stats = middleware_stat () in
|
||||||
|
Server.add_middleware server ~stage:(`Stage 1) m_stats;
|
||||||
|
|
||||||
|
(* say hello *)
|
||||||
|
Server.add_route_handler ~meth:`GET server
|
||||||
|
Route.(exact "hello" @/ string @/ return)
|
||||||
|
(fun name _req -> Response.make_string (Ok ("hello " ^ name ^ "!\n")));
|
||||||
|
|
||||||
|
(* compressed file access *)
|
||||||
|
Server.add_route_handler ~meth:`GET server
|
||||||
|
Route.(exact "zcat" @/ string_urlencoded @/ return)
|
||||||
|
(fun path _req ->
|
||||||
|
let ic = open_in path in
|
||||||
|
let str = IO.Input.of_in_channel ic in
|
||||||
|
let mime_type =
|
||||||
|
try
|
||||||
|
let p = Unix.open_process_in (Printf.sprintf "file -i -b %S" path) in
|
||||||
|
try
|
||||||
|
let s = [ "Content-Type", String.trim (input_line p) ] in
|
||||||
|
ignore @@ Unix.close_process_in p;
|
||||||
|
s
|
||||||
|
with _ ->
|
||||||
|
ignore @@ Unix.close_process_in p;
|
||||||
|
[]
|
||||||
|
with _ -> []
|
||||||
|
in
|
||||||
|
Response.make_stream ~headers:mime_type (Ok str));
|
||||||
|
|
||||||
|
(* echo request *)
|
||||||
|
Server.add_route_handler server
|
||||||
|
Route.(exact "echo" @/ return)
|
||||||
|
(fun req ->
|
||||||
|
let q =
|
||||||
|
Request.query req
|
||||||
|
|> List.map (fun (k, v) -> Printf.sprintf "%S = %S" k v)
|
||||||
|
|> String.concat ";"
|
||||||
|
in
|
||||||
|
Response.make_string
|
||||||
|
(Ok (Format.asprintf "echo:@ %a@ (query: %s)@." Request.pp req q)));
|
||||||
|
|
||||||
|
(* file upload *)
|
||||||
|
Server.add_route_handler_stream ~meth:`PUT server
|
||||||
|
Route.(exact "upload" @/ string @/ return)
|
||||||
|
(fun path req ->
|
||||||
|
Log.debug (fun k ->
|
||||||
|
k "start upload %S, headers:\n%s\n\n%!" path
|
||||||
|
(Format.asprintf "%a" Headers.pp (Request.headers req)));
|
||||||
|
try
|
||||||
|
let oc = open_out @@ "/tmp/" ^ path in
|
||||||
|
IO.Input.to_chan oc req.Request.body;
|
||||||
|
flush oc;
|
||||||
|
Response.make_string (Ok "uploaded file")
|
||||||
|
with e ->
|
||||||
|
Response.fail ~code:500 "couldn't upload file: %s"
|
||||||
|
(Printexc.to_string e));
|
||||||
|
|
||||||
|
(* protected by login *)
|
||||||
|
Server.add_route_handler server
|
||||||
|
Route.(exact "protected" @/ return)
|
||||||
|
(fun req ->
|
||||||
|
let ok =
|
||||||
|
match Request.get_header req "authorization" with
|
||||||
|
| Some v ->
|
||||||
|
Log.debug (fun k -> k "authenticate with %S" v);
|
||||||
|
v = "Basic " ^ base64 "user:foobar"
|
||||||
|
| None -> false
|
||||||
|
in
|
||||||
|
if ok then (
|
||||||
|
(* FIXME: a logout link *)
|
||||||
|
let s =
|
||||||
|
"<p>hello, this is super secret!</p><a href=\"/logout\">log out</a>"
|
||||||
|
in
|
||||||
|
Response.make_string (Ok s)
|
||||||
|
) else (
|
||||||
|
let headers =
|
||||||
|
Headers.(empty |> set "www-authenticate" "basic realm=\"echo\"")
|
||||||
|
in
|
||||||
|
Response.fail ~code:401 ~headers "invalid"
|
||||||
|
));
|
||||||
|
|
||||||
|
(* logout *)
|
||||||
|
Server.add_route_handler server
|
||||||
|
Route.(exact "logout" @/ return)
|
||||||
|
(fun _req -> Response.fail ~code:401 "logged out");
|
||||||
|
|
||||||
|
(* stats *)
|
||||||
|
Server.add_route_handler server
|
||||||
|
Route.(exact "stats" @/ return)
|
||||||
|
(fun _req ->
|
||||||
|
let stats = get_stats () in
|
||||||
|
Response.make_string @@ Ok stats);
|
||||||
|
|
||||||
|
Server.add_route_handler server
|
||||||
|
Route.(exact "alice" @/ return)
|
||||||
|
(fun _req -> Response.make_string (Ok alice_text));
|
||||||
|
|
||||||
|
(* main page *)
|
||||||
|
Server.add_route_handler server
|
||||||
|
Route.(return)
|
||||||
|
(fun _req ->
|
||||||
|
let open Tiny_httpd_html in
|
||||||
|
let h =
|
||||||
|
html []
|
||||||
|
[
|
||||||
|
head [] [ title [] [ txt "index of echo" ] ];
|
||||||
|
body []
|
||||||
|
[
|
||||||
|
h3 [] [ txt "welcome!" ];
|
||||||
|
p [] [ b [] [ txt "endpoints are:" ] ];
|
||||||
|
ul []
|
||||||
|
[
|
||||||
|
li [] [ pre [] [ txt "/hello/:name (GET)" ] ];
|
||||||
|
li []
|
||||||
|
[
|
||||||
|
pre []
|
||||||
|
[
|
||||||
|
a [ A.href "/echo/" ] [ txt "echo" ];
|
||||||
|
txt " echo back query";
|
||||||
|
];
|
||||||
|
];
|
||||||
|
li []
|
||||||
|
[ pre [] [ txt "/upload/:path (PUT) to upload a file" ] ];
|
||||||
|
li []
|
||||||
|
[
|
||||||
|
pre []
|
||||||
|
[
|
||||||
|
txt
|
||||||
|
"/zcat/:path (GET) to download a file (deflate \
|
||||||
|
transfer-encoding)";
|
||||||
|
];
|
||||||
|
];
|
||||||
|
li []
|
||||||
|
[
|
||||||
|
pre []
|
||||||
|
[
|
||||||
|
a [ A.href "/stats/" ] [ txt "/stats/" ];
|
||||||
|
txt " (GET) to access statistics";
|
||||||
|
];
|
||||||
|
];
|
||||||
|
li []
|
||||||
|
[
|
||||||
|
pre []
|
||||||
|
[
|
||||||
|
a [ A.href "/vfs/" ] [ txt "/vfs" ];
|
||||||
|
txt " (GET) to access a VFS embedded in the binary";
|
||||||
|
];
|
||||||
|
];
|
||||||
|
li []
|
||||||
|
[
|
||||||
|
pre []
|
||||||
|
[
|
||||||
|
a [ A.href "/protected" ] [ txt "/protected" ];
|
||||||
|
txt
|
||||||
|
" (GET) to see a protected page (login: user, \
|
||||||
|
password: foobar)";
|
||||||
|
];
|
||||||
|
];
|
||||||
|
li []
|
||||||
|
[
|
||||||
|
pre []
|
||||||
|
[
|
||||||
|
a [ A.href "/logout" ] [ txt "/logout" ];
|
||||||
|
txt " (POST) to log out";
|
||||||
|
];
|
||||||
|
];
|
||||||
|
];
|
||||||
|
];
|
||||||
|
]
|
||||||
|
in
|
||||||
|
let s = to_string_top h in
|
||||||
|
Response.make_string ~headers:[ "content-type", "text/html" ] @@ Ok s);
|
||||||
|
|
||||||
|
Printf.printf "listening on http://%s:%d\n%!" (Server.addr server)
|
||||||
|
(Server.port server);
|
||||||
|
match Server.run server with
|
||||||
|
| Ok () -> ()
|
||||||
|
| Error e -> raise e
|
||||||
21
tests/dune
21
tests/dune
|
|
@ -19,6 +19,27 @@
|
||||||
(action
|
(action
|
||||||
(diff echo1.expect echo1.out)))
|
(diff echo1.expect echo1.out)))
|
||||||
|
|
||||||
|
(rule
|
||||||
|
(targets echo_mio1.out)
|
||||||
|
(deps
|
||||||
|
(:bin ../examples/echo_mio.exe))
|
||||||
|
(locks /port)
|
||||||
|
(enabled_if
|
||||||
|
(= %{system} "linux"))
|
||||||
|
(package tiny_httpd_moonpool)
|
||||||
|
(action
|
||||||
|
(with-stdout-to
|
||||||
|
%{targets}
|
||||||
|
(run ./echo_mio1.sh %{bin}))))
|
||||||
|
|
||||||
|
(rule
|
||||||
|
(alias runtest)
|
||||||
|
(package tiny_httpd_moonpool)
|
||||||
|
(enabled_if
|
||||||
|
(= %{system} "linux"))
|
||||||
|
(action
|
||||||
|
(diff echo_mio1.expect echo_mio1.out)))
|
||||||
|
|
||||||
(rule
|
(rule
|
||||||
(targets sse_count.out)
|
(targets sse_count.out)
|
||||||
(deps
|
(deps
|
||||||
|
|
|
||||||
10
tests/echo_mio1.expect
Normal file
10
tests/echo_mio1.expect
Normal file
|
|
@ -0,0 +1,10 @@
|
||||||
|
listening on http://127.0.0.1:8085
|
||||||
|
test moonpool_io
|
||||||
|
echo:
|
||||||
|
{meth=GET; host=localhost:8085;
|
||||||
|
headers=[user-agent: test
|
||||||
|
accept: */*
|
||||||
|
host: localhost:8085];
|
||||||
|
path="/echo/?a=b&c=d"; body=""; path_components=["echo"];
|
||||||
|
query=["c","d";"a","b"]}
|
||||||
|
(query: "c" = "d";"a" = "b")
|
||||||
12
tests/echo_mio1.sh
Executable file
12
tests/echo_mio1.sh
Executable file
|
|
@ -0,0 +1,12 @@
|
||||||
|
#!/usr/bin/env sh
|
||||||
|
|
||||||
|
ECHO=$1
|
||||||
|
PORT=8085
|
||||||
|
|
||||||
|
"$ECHO" -p $PORT &
|
||||||
|
PID=$!
|
||||||
|
sleep 0.1
|
||||||
|
echo "test moonpool_io"
|
||||||
|
curl -N "http://localhost:${PORT}/echo/?a=b&c=d" -H user-agent:test --max-time 5
|
||||||
|
|
||||||
|
kill $PID
|
||||||
Loading…
Add table
Reference in a new issue