mirror of
				https://github.com/jart/cosmopolitan.git
				synced 2025-10-25 10:40:57 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			509 lines
		
	
	
	
		
			15 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			509 lines
		
	
	
	
		
			15 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
 | |
| │vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8                                :vi│
 | |
| ╞══════════════════════════════════════════════════════════════════════════════╡
 | |
| │ Copyright 2021 Justine Alexandra Roberts Tunney                              │
 | |
| │                                                                              │
 | |
| │ Permission to use, copy, modify, and/or distribute this software for         │
 | |
| │ any purpose with or without fee is hereby granted, provided that the         │
 | |
| │ above copyright notice and this permission notice appear in all copies.      │
 | |
| │                                                                              │
 | |
| │ THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL                │
 | |
| │ WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED                │
 | |
| │ WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE             │
 | |
| │ AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL         │
 | |
| │ DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR        │
 | |
| │ PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER               │
 | |
| │ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR             │
 | |
| │ PERFORMANCE OF THIS SOFTWARE.                                                │
 | |
| ╚─────────────────────────────────────────────────────────────────────────────*/
 | |
| #include "libc/assert.h"
 | |
| #include "libc/calls/calls.h"
 | |
| #include "libc/dns/dns.h"
 | |
| #include "libc/errno.h"
 | |
| #include "libc/fmt/conv.h"
 | |
| #include "libc/log/check.h"
 | |
| #include "libc/log/log.h"
 | |
| #include "libc/macros.internal.h"
 | |
| #include "libc/math.h"
 | |
| #include "libc/mem/mem.h"
 | |
| #include "libc/stdio/rand.h"
 | |
| #include "libc/runtime/gc.internal.h"
 | |
| #include "libc/sock/goodsocket.internal.h"
 | |
| #include "libc/sock/sock.h"
 | |
| #include "libc/stdio/append.internal.h"
 | |
| #include "libc/stdio/stdio.h"
 | |
| #include "libc/str/slice.h"
 | |
| #include "libc/str/str.h"
 | |
| #include "libc/sysv/consts/af.h"
 | |
| #include "libc/sysv/consts/ex.h"
 | |
| #include "libc/sysv/consts/exit.h"
 | |
| #include "libc/sysv/consts/ipproto.h"
 | |
| #include "libc/sysv/consts/sig.h"
 | |
| #include "libc/sysv/consts/so.h"
 | |
| #include "libc/sysv/consts/sock.h"
 | |
| #include "libc/sysv/consts/sol.h"
 | |
| #include "libc/sysv/consts/tcp.h"
 | |
| #include "libc/time/time.h"
 | |
| #include "libc/x/x.h"
 | |
| #include "net/http/http.h"
 | |
| #include "net/http/url.h"
 | |
| #include "net/https/https.h"
 | |
| #include "third_party/getopt/getopt.h"
 | |
| #include "third_party/mbedtls/ctr_drbg.h"
 | |
| #include "third_party/mbedtls/debug.h"
 | |
| #include "third_party/mbedtls/error.h"
 | |
| #include "third_party/mbedtls/net_sockets.h"
 | |
| #include "third_party/mbedtls/ssl.h"
 | |
| 
 | |
| #define OPTS "BIqksvzX:H:C:m:"
 | |
| 
 | |
| #define Micros(t)       ((int64_t)((t)*1e6))
 | |
| #define HasHeader(H)    (!!msg.headers[H].a)
 | |
| #define HeaderData(H)   (inbuf.p + msg.headers[H].a)
 | |
| #define HeaderLength(H) (msg.headers[H].b - msg.headers[H].a)
 | |
| #define HeaderEqualCase(H, S) \
 | |
|   SlicesEqualCase(S, strlen(S), HeaderData(H), HeaderLength(H))
 | |
| 
 | |
| struct Buffer {
 | |
|   size_t n, c;
 | |
|   char *p;
 | |
| };
 | |
| 
 | |
| struct Headers {
 | |
|   size_t n;
 | |
|   char **p;
 | |
| } headers;
 | |
| 
 | |
| bool suiteb;
 | |
| char *request;
 | |
| bool isdone;
 | |
| char *urlarg;
 | |
| int method = kHttpGet;
 | |
| bool authmode = MBEDTLS_SSL_VERIFY_NONE;
 | |
| 
 | |
| char *host;
 | |
| char *port;
 | |
| char *flags;
 | |
| bool usessl;
 | |
| uint32_t ip;
 | |
| struct Url url;
 | |
| struct addrinfo *addr;
 | |
| struct Buffer inbuf;
 | |
| 
 | |
| long error_count;
 | |
| long failure_count;
 | |
| long message_count;
 | |
| long connect_count;
 | |
| double *latencies;
 | |
| size_t latencies_n;
 | |
| size_t latencies_c;
 | |
| long double start_run;
 | |
| long double end_run;
 | |
| long double start_fetch;
 | |
| long double end_fetch;
 | |
| long connectionstobemade = 100;
 | |
| long messagesperconnection = 100;
 | |
| 
 | |
| mbedtls_x509_crt *cachain;
 | |
| mbedtls_ssl_config conf;
 | |
| mbedtls_ssl_context ssl;
 | |
| mbedtls_ctr_drbg_context drbg;
 | |
| 
 | |
| struct addrinfo hints = {.ai_family = AF_INET,
 | |
|                          .ai_socktype = SOCK_STREAM,
 | |
|                          .ai_protocol = IPPROTO_TCP,
 | |
|                          .ai_flags = AI_NUMERICSERV};
 | |
| 
 | |
| void OnInt(int sig) {
 | |
|   isdone = true;
 | |
| }
 | |
| 
 | |
| static int TlsSend(void *c, const unsigned char *p, size_t n) {
 | |
|   int rc;
 | |
|   if ((rc = write(*(int *)c, p, n)) == -1) {
 | |
|     if (errno == EINTR) {
 | |
|       return MBEDTLS_ERR_SSL_WANT_WRITE;
 | |
|     } else if (errno == EAGAIN) {
 | |
|       return MBEDTLS_ERR_SSL_TIMEOUT;
 | |
|     } else if (errno == EPIPE || errno == ECONNRESET || errno == ENETRESET) {
 | |
|       return MBEDTLS_ERR_NET_CONN_RESET;
 | |
|     } else {
 | |
|       VERBOSEF("tls write() error %s", strerror(errno));
 | |
|       return MBEDTLS_ERR_NET_RECV_FAILED;
 | |
|     }
 | |
|   }
 | |
|   return rc;
 | |
| }
 | |
| 
 | |
| static int TlsRecv(void *c, unsigned char *p, size_t n, uint32_t o) {
 | |
|   int r;
 | |
|   if ((r = read(*(int *)c, p, n)) == -1) {
 | |
|     if (errno == EINTR) {
 | |
|       return MBEDTLS_ERR_SSL_WANT_READ;
 | |
|     } else if (errno == EAGAIN) {
 | |
|       return MBEDTLS_ERR_SSL_TIMEOUT;
 | |
|     } else if (errno == EPIPE || errno == ECONNRESET || errno == ENETRESET) {
 | |
|       return MBEDTLS_ERR_NET_CONN_RESET;
 | |
|     } else {
 | |
|       VERBOSEF("tls read() error %s", strerror(errno));
 | |
|       return MBEDTLS_ERR_NET_RECV_FAILED;
 | |
|     }
 | |
|   }
 | |
|   return r;
 | |
| }
 | |
| 
 | |
| static wontreturn void PrintUsage(FILE *f, int rc) {
 | |
|   fprintf(f, "usage: %s [-%s] URL\n", OPTS, program_invocation_name);
 | |
|   fprintf(f, "wb - cosmopolitan http/https benchmark tool\n");
 | |
|   fprintf(f, "  -C INT   connections to be made\n");
 | |
|   fprintf(f, "  -m INT   messages per connection\n");
 | |
|   fprintf(f, "  -B       use suite b ciphersuites\n");
 | |
|   fprintf(f, "  -v       increase verbosity\n");
 | |
|   fprintf(f, "  -H K:V   append http header\n");
 | |
|   fprintf(f, "  -X NAME  specify http method\n");
 | |
|   fprintf(f, "  -k       verify ssl certs\n");
 | |
|   fprintf(f, "  -I       same as -X HEAD\n");
 | |
|   fprintf(f, "  -z       same as -H Accept-Encoding:gzip\n");
 | |
|   fprintf(f, "  -h       show this help\n");
 | |
|   exit(rc);
 | |
| }
 | |
| 
 | |
| int fetch(void) {
 | |
|   char *p;
 | |
|   int status;
 | |
|   ssize_t rc;
 | |
|   const char *body;
 | |
|   int t, ret, sock;
 | |
|   struct TlsBio *bio;
 | |
|   long messagesremaining;
 | |
|   struct HttpMessage msg;
 | |
|   struct HttpUnchunker u;
 | |
|   size_t urlarglen, requestlen;
 | |
|   size_t g, i, n, hdrsize, paylen;
 | |
| 
 | |
|   messagesremaining = messagesperconnection;
 | |
| 
 | |
|   /*
 | |
|    * Setup crypto.
 | |
|    */
 | |
|   if (usessl) {
 | |
|     -mbedtls_ssl_session_reset(&ssl);
 | |
|     CHECK_EQ(0, mbedtls_ssl_set_hostname(&ssl, host));
 | |
|   }
 | |
| 
 | |
|   /*
 | |
|    * Connect to server.
 | |
|    */
 | |
|   InitHttpMessage(&msg, kHttpResponse);
 | |
|   ip = ntohl(((struct sockaddr_in *)addr->ai_addr)->sin_addr.s_addr);
 | |
|   CHECK_NE(-1, (sock = GoodSocket(addr->ai_family, addr->ai_socktype,
 | |
|                                   addr->ai_protocol, false, 0)));
 | |
|   if (connect(sock, addr->ai_addr, addr->ai_addrlen) == -1) {
 | |
|     goto TransportError;
 | |
|   }
 | |
|   if (usessl) {
 | |
|     mbedtls_ssl_set_bio(&ssl, &sock, TlsSend, 0, TlsRecv);
 | |
|     if ((ret = mbedtls_ssl_handshake(&ssl))) {
 | |
|       goto TransportError;
 | |
|     }
 | |
|   }
 | |
| 
 | |
| SendAnother:
 | |
| 
 | |
|   /*
 | |
|    * Send HTTP Message.
 | |
|    */
 | |
|   n = appendz(request).i;
 | |
|   if (usessl) {
 | |
|     ret = mbedtls_ssl_write(&ssl, request, n);
 | |
|     if (ret != n) goto TransportError;
 | |
|   } else if (write(sock, request, n) != n) {
 | |
|     goto TransportError;
 | |
|   }
 | |
| 
 | |
|   /*
 | |
|    * Handle response.
 | |
|    */
 | |
|   InitHttpMessage(&msg, kHttpResponse);
 | |
|   for (hdrsize = paylen = t = 0;;) {
 | |
|     if (inbuf.n == inbuf.c) {
 | |
|       inbuf.c += 1000;
 | |
|       inbuf.c += inbuf.c >> 1;
 | |
|       inbuf.p = realloc(inbuf.p, inbuf.c);
 | |
|     }
 | |
|     if (usessl) {
 | |
|       if ((rc = mbedtls_ssl_read(&ssl, inbuf.p + inbuf.n, inbuf.c - inbuf.n)) <
 | |
|           0) {
 | |
|         if (rc == MBEDTLS_ERR_SSL_PEER_CLOSE_NOTIFY) {
 | |
|           rc = 0;
 | |
|         } else {
 | |
|           goto TransportError;
 | |
|         }
 | |
|       }
 | |
|     } else if ((rc = read(sock, inbuf.p + inbuf.n, inbuf.c - inbuf.n)) == -1) {
 | |
|       goto TransportError;
 | |
|     }
 | |
|     g = rc;
 | |
|     inbuf.n += g;
 | |
|     switch (t) {
 | |
|       case kHttpClientStateHeaders:
 | |
|         if (!g) goto TransportError;
 | |
|         rc = ParseHttpMessage(&msg, inbuf.p, inbuf.n);
 | |
|         if (rc == -1) goto TransportError;
 | |
|         if (rc) {
 | |
|           hdrsize = rc;
 | |
|           if (100 <= msg.status && msg.status <= 199) {
 | |
|             if ((HasHeader(kHttpContentLength) &&
 | |
|                  !HeaderEqualCase(kHttpContentLength, "0")) ||
 | |
|                 (HasHeader(kHttpTransferEncoding) &&
 | |
|                  !HeaderEqualCase(kHttpTransferEncoding, "identity"))) {
 | |
|               goto TransportError;
 | |
|             }
 | |
|             DestroyHttpMessage(&msg);
 | |
|             InitHttpMessage(&msg, kHttpResponse);
 | |
|             memmove(inbuf.p, inbuf.p + hdrsize, inbuf.n - hdrsize);
 | |
|             inbuf.n -= hdrsize;
 | |
|             break;
 | |
|           }
 | |
|           if (msg.status == 204 || msg.status == 304) {
 | |
|             goto Finished;
 | |
|           }
 | |
|           if (HasHeader(kHttpTransferEncoding) &&
 | |
|               !HeaderEqualCase(kHttpTransferEncoding, "identity")) {
 | |
|             if (HeaderEqualCase(kHttpTransferEncoding, "chunked")) {
 | |
|               t = kHttpClientStateBodyChunked;
 | |
|               bzero(&u, sizeof(u));
 | |
|               goto Chunked;
 | |
|             } else {
 | |
|               goto TransportError;
 | |
|             }
 | |
|           } else if (HasHeader(kHttpContentLength)) {
 | |
|             rc = ParseContentLength(HeaderData(kHttpContentLength),
 | |
|                                     HeaderLength(kHttpContentLength));
 | |
|             if (rc == -1) goto TransportError;
 | |
|             if ((paylen = rc) <= inbuf.n - hdrsize) {
 | |
|               goto Finished;
 | |
|             } else {
 | |
|               t = kHttpClientStateBodyLengthed;
 | |
|             }
 | |
|           } else {
 | |
|             t = kHttpClientStateBody;
 | |
|           }
 | |
|         }
 | |
|         break;
 | |
|       case kHttpClientStateBody:
 | |
|         if (!g) {
 | |
|           paylen = inbuf.n;
 | |
|           goto Finished;
 | |
|         }
 | |
|         break;
 | |
|       case kHttpClientStateBodyLengthed:
 | |
|         if (!g) goto TransportError;
 | |
|         if (inbuf.n - hdrsize >= paylen) goto Finished;
 | |
|         break;
 | |
|       case kHttpClientStateBodyChunked:
 | |
|       Chunked:
 | |
|         rc = Unchunk(&u, inbuf.p + hdrsize, inbuf.n - hdrsize, &paylen);
 | |
|         if (rc == -1) goto TransportError;
 | |
|         if (rc) goto Finished;
 | |
|         break;
 | |
|       default:
 | |
|         unreachable;
 | |
|     }
 | |
|   }
 | |
| 
 | |
| Finished:
 | |
|   status = msg.status;
 | |
|   DestroyHttpMessage(&msg);
 | |
|   if (!isdone && status == 200 && --messagesremaining > 0) {
 | |
|     long double now = nowl();
 | |
|     end_fetch = now;
 | |
|     ++message_count;
 | |
|     latencies = realloc(latencies, ++latencies_n * sizeof(*latencies));
 | |
|     latencies[latencies_n - 1] = end_fetch - start_fetch;
 | |
|     start_fetch = now;
 | |
|     goto SendAnother;
 | |
|   }
 | |
|   close(sock);
 | |
|   return status;
 | |
| TransportError:
 | |
|   close(sock);
 | |
|   DestroyHttpMessage(&msg);
 | |
|   return 900;
 | |
| }
 | |
| 
 | |
| int main(int argc, char *argv[]) {
 | |
|   xsigaction(SIGPIPE, SIG_IGN, 0, 0, 0);
 | |
|   xsigaction(SIGINT, OnInt, 0, 0, 0);
 | |
| 
 | |
|   /*
 | |
|    * Read flags.
 | |
|    */
 | |
|   int opt;
 | |
|   __log_level = kLogWarn;
 | |
|   while ((opt = getopt(argc, argv, OPTS)) != -1) {
 | |
|     switch (opt) {
 | |
|       case 's':
 | |
|       case 'q':
 | |
|         break;
 | |
|       case 'B':
 | |
|         suiteb = true;
 | |
|         appendf(&flags, " -B");
 | |
|         break;
 | |
|       case 'v':
 | |
|         ++__log_level;
 | |
|         break;
 | |
|       case 'I':
 | |
|         method = kHttpHead;
 | |
|         appendf(&flags, " -I");
 | |
|         break;
 | |
|       case 'H':
 | |
|         headers.p = realloc(headers.p, ++headers.n * sizeof(*headers.p));
 | |
|         headers.p[headers.n - 1] = optarg;
 | |
|         appendf(&flags, " -H '%s'", optarg);
 | |
|         break;
 | |
|       case 'z':
 | |
|         headers.p = realloc(headers.p, ++headers.n * sizeof(*headers.p));
 | |
|         headers.p[headers.n - 1] = "Accept-Encoding: gzip";
 | |
|         appendf(&flags, " -z");
 | |
|         break;
 | |
|       case 'X':
 | |
|         CHECK((method = GetHttpMethod(optarg, strlen(optarg))));
 | |
|         appendf(&flags, " -X %s", optarg);
 | |
|         break;
 | |
|       case 'k':
 | |
|         authmode = MBEDTLS_SSL_VERIFY_REQUIRED;
 | |
|         appendf(&flags, " -k");
 | |
|         break;
 | |
|       case 'm':
 | |
|         messagesperconnection = strtol(optarg, 0, 0);
 | |
|         break;
 | |
|       case 'C':
 | |
|         connectionstobemade = strtol(optarg, 0, 0);
 | |
|         break;
 | |
|       case 'h':
 | |
|         PrintUsage(stdout, EXIT_SUCCESS);
 | |
|       default:
 | |
|         PrintUsage(stderr, EX_USAGE);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   appendf(&flags, " -m %ld", messagesperconnection);
 | |
|   appendf(&flags, " -C %ld", connectionstobemade);
 | |
| 
 | |
|   if (optind == argc) PrintUsage(stdout, EXIT_SUCCESS);
 | |
|   urlarg = argv[optind];
 | |
|   cachain = GetSslRoots();
 | |
| 
 | |
|   long connectsremaining = connectionstobemade;
 | |
| 
 | |
|   /*
 | |
|    * Parse URL.
 | |
|    */
 | |
|   gc(ParseUrl(urlarg, -1, &url));
 | |
|   gc(url.params.p);
 | |
|   usessl = false;
 | |
|   if (url.scheme.n) {
 | |
|     if (url.scheme.n == 5 && !memcasecmp(url.scheme.p, "https", 5)) {
 | |
|       usessl = true;
 | |
|     } else if (!(url.scheme.n == 4 && !memcasecmp(url.scheme.p, "http", 4))) {
 | |
|       FATALF("bad scheme");
 | |
|     }
 | |
|   }
 | |
|   if (url.host.n) {
 | |
|     host = gc(strndup(url.host.p, url.host.n));
 | |
|     if (url.port.n) {
 | |
|       port = gc(strndup(url.port.p, url.port.n));
 | |
|     } else {
 | |
|       port = usessl ? "443" : "80";
 | |
|     }
 | |
|   } else {
 | |
|     host = "127.0.0.1";
 | |
|     port = "80";
 | |
|   }
 | |
|   CHECK(IsAcceptableHost(host, -1));
 | |
|   url.fragment.p = 0, url.fragment.n = 0;
 | |
|   url.scheme.p = 0, url.scheme.n = 0;
 | |
|   url.user.p = 0, url.user.n = 0;
 | |
|   url.pass.p = 0, url.pass.n = 0;
 | |
|   url.host.p = 0, url.host.n = 0;
 | |
|   url.port.p = 0, url.port.n = 0;
 | |
|   if (!url.path.n || url.path.p[0] != '/') {
 | |
|     char *p = gc(xmalloc(1 + url.path.n));
 | |
|     mempcpy(mempcpy(p, "/", 1), url.path.p, url.path.n);
 | |
|     url.path.p = p;
 | |
|     ++url.path.n;
 | |
|   }
 | |
| 
 | |
|   /*
 | |
|    * Create HTTP message.
 | |
|    */
 | |
|   appendf(&request,
 | |
|           "%s %s HTTP/1.1\r\n"
 | |
|           "Host: %s:%s\r\n",
 | |
|           kHttpMethod[method], _gc(EncodeUrl(&url, 0)), host, port);
 | |
|   for (int i = 0; i < headers.n; ++i) {
 | |
|     appendf(&request, "%s\r\n", headers.p[i]);
 | |
|   }
 | |
|   appendf(&request, "\r\n");
 | |
| 
 | |
|   /*
 | |
|    * Perform DNS lookup.
 | |
|    */
 | |
|   int rc;
 | |
|   if ((rc = getaddrinfo(host, port, &hints, &addr)) != EAI_SUCCESS) {
 | |
|     FATALF("getaddrinfo(%s:%s) failed", host, port);
 | |
|   }
 | |
| 
 | |
|   /*
 | |
|    * Setup SSL crypto.
 | |
|    */
 | |
|   mbedtls_ssl_init(&ssl);
 | |
|   mbedtls_ctr_drbg_init(&drbg);
 | |
|   mbedtls_ssl_config_init(&conf);
 | |
|   CHECK_EQ(0, mbedtls_ctr_drbg_seed(&drbg, GetEntropy, 0, "justine", 7));
 | |
|   CHECK_EQ(0,
 | |
|            mbedtls_ssl_config_defaults(
 | |
|                &conf, MBEDTLS_SSL_IS_CLIENT, MBEDTLS_SSL_TRANSPORT_STREAM,
 | |
|                suiteb ? MBEDTLS_SSL_PRESET_SUITEB : MBEDTLS_SSL_PRESET_SUITEC));
 | |
|   mbedtls_ssl_conf_authmode(&conf, authmode);
 | |
|   mbedtls_ssl_conf_ca_chain(&conf, cachain, 0);
 | |
|   mbedtls_ssl_conf_rng(&conf, mbedtls_ctr_drbg_random, &drbg);
 | |
|   CHECK_EQ(0, mbedtls_ssl_setup(&ssl, &conf));
 | |
| 
 | |
|   int status;
 | |
|   latencies_c = 1024;
 | |
|   latencies = malloc(latencies_c * sizeof(*latencies));
 | |
|   start_run = nowl();
 | |
|   while (!isdone && --connectsremaining >= 0) {
 | |
|     start_fetch = nowl();
 | |
|     status = fetch();
 | |
|     end_fetch = nowl();
 | |
|     if (status == 200) {
 | |
|       ++connect_count;
 | |
|       ++message_count;
 | |
|       latencies = realloc(latencies, ++latencies_n * sizeof(*latencies));
 | |
|       latencies[latencies_n - 1] = end_fetch - start_fetch;
 | |
|     } else if (status == 900) {
 | |
|       ++failure_count;
 | |
|     } else {
 | |
|       ++error_count;
 | |
|     }
 | |
|   }
 | |
|   end_run = nowl();
 | |
| 
 | |
|   double latencies_sum = fsum(latencies, latencies_n);
 | |
|   double avg_latency = latencies_sum / message_count;
 | |
| 
 | |
|   printf("wb%s\n", flags);
 | |
|   printf("msgs / second:   %,ld qps\n",
 | |
|          (int64_t)(message_count / (end_run - start_run)));
 | |
|   printf("run time:        %,ldµs\n", Micros(end_run - start_run));
 | |
|   printf("latency / msgs:  %,ldµs\n", Micros(avg_latency));
 | |
|   printf("message count:   %,ld\n", message_count);
 | |
|   printf("connect count:   %,ld\n", connect_count);
 | |
|   printf("error count:     %,ld (non-200 responses)\n", error_count);
 | |
|   printf("failure count:   %,ld (transport error)\n", failure_count);
 | |
| 
 | |
|   return 0;
 | |
| }
 |