File: bench_http.ml

package info (click to toggle)
ocaml-eio 1.3-2
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 2,548 kB
  • sloc: ml: 14,608; ansic: 1,237; makefile: 25
file content (107 lines) | stat: -rw-r--r-- 4,091 bytes parent folder | download
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
(* A multi-domain server handles HTTP-like requests from many clients running across multiple domains. *)

open Eio.Std

(* Note: this is not a real HTTP parser! *)
let key_char = function
  | 'A'..'Z' | 'a'..'z' | '-' -> true
  | _ -> false

let parse_headers r =
  let len = ref (-1) in
  let rec aux () =
    let key = Eio.Buf_read.take_while key_char r in
    if key = "" then Eio.Buf_read.string "\r\n" r
    else (
      Eio.Buf_read.char ':' r;
      let value = Eio.Buf_read.line r in
      if key = "Content-Length" then len := int_of_string (String.trim value);
      aux ()
    )
  in
  aux ();
  !len

let handle_connection conn _addr =
  Eio.Buf_write.with_flow conn @@ fun w ->
  let rec requests r =
    let _req = Eio.Buf_read.line r in
    let len = parse_headers r in
    let body = Eio.Buf_read.take len r in
    let response = body ^ " / received" in
    Eio.Buf_write.string w "HTTP/1.1 200 OK\r\n";
    Eio.Buf_write.string w (Printf.sprintf "Content-Length: %d\r\n" (String.length response));
    Eio.Buf_write.string w "\r\n";
    Eio.Buf_write.string w response;
    if not (Eio.Buf_read.at_end_of_input r) then requests r
  in
  Eio.Buf_read.parse_exn requests conn ~max_size:max_int

let run_client ~n_requests id conn =
  let total = ref 0 in
  let r = Eio.Buf_read.of_flow conn ~max_size:max_int in
  Eio.Buf_write.with_flow conn @@ fun w ->
  for i = 1 to n_requests do
    let msg = Printf.sprintf "%s / request %d" id i in
    Eio.Buf_write.string w "POST / HTTP/1.1\r\n";
    Eio.Buf_write.string w "Host: localhost:8085\r\n";
    Eio.Buf_write.string w "User-Agent: bench_server\r\n";
    Eio.Buf_write.string w "Connection: keep-alive\r\n";
    Eio.Buf_write.string w (Printf.sprintf "Content-Length: %d\r\n" (String.length msg));
    Eio.Buf_write.string w "\r\n";
    Eio.Buf_write.string w msg;
    let status = Eio.Buf_read.line r in
    assert (status = "HTTP/1.1 200 OK");
    let len = parse_headers r in
    let body = Eio.Buf_read.take len r in
    assert (body = msg ^ " / received");
    incr total
  done;
  !total

let main net domain_mgr ~n_client_domains ~n_server_domains ~n_connections_per_domain ~n_requests_per_connection =
  let total = Atomic.make 0 in
  let t0 = Unix.gettimeofday () in
  Switch.run ~name:"main" (fun sw ->
      let addr = `Tcp (Eio.Net.Ipaddr.V4.loopback, 8085) in
      let backlog = n_connections_per_domain * n_client_domains in
      let server_socket = Eio.Net.listen ~reuse_addr:true ~backlog ~sw net addr in
      Fiber.fork_daemon ~sw (fun () ->
          Eio.Net.run_server server_socket handle_connection
            ~additional_domains:(domain_mgr, n_server_domains - 1)
            ~on_error:raise
        );
      for domain = 1 to n_client_domains do
        Fiber.fork ~sw (fun () ->
            Eio.Domain_manager.run domain_mgr (fun () ->
                Switch.run ~name:"client-domain" @@ fun sw ->
                for i = 1 to n_connections_per_domain do
                  Fiber.fork ~sw (fun () ->
                      let id = Printf.sprintf "domain %d / conn %d" domain i in
                      let conn = Eio.Net.connect ~sw net addr in
                      let requests = run_client ~n_requests:n_requests_per_connection id conn in
                      ignore (Atomic.fetch_and_add total requests : int)
                    )
                done
              )
          )
      done
    );
  let t1 = Unix.gettimeofday () in
  (* Fmt.pr "clients, servers, requests, requests/s@."; *)
  let requests = n_connections_per_domain * n_client_domains * n_requests_per_connection in
  assert (requests = Atomic.get total);
  let req_per_s = float requests /. (t1 -. t0) in
  Metric.create
    (Printf.sprintf "requests:%d client-domains:%d server-domains:%d" requests n_client_domains n_server_domains)
   (`Float req_per_s) "requests/s" "Request rate of a HTTP client/server system"

let run env =
  let metrics =
    main env#net env#domain_mgr
      ~n_client_domains:4
      ~n_server_domains:4
      ~n_connections_per_domain:25
      ~n_requests_per_connection:1000
  in
  [metrics]