summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorVsevolod Stakhov <vsevolod@highsecure.ru>2015-10-20 19:10:19 +0100
committerVsevolod Stakhov <vsevolod@highsecure.ru>2015-10-20 19:10:19 +0100
commit1027e57ab831916be8bd5ae678e667ca93dffe43 (patch)
treeb196d84be62119e0e70b93813eacf28cff00edc1
parentef10b624276f126a42fd6a44ecb1d889753121b9 (diff)
downloadrspamd-1027e57ab831916be8bd5ae678e667ca93dffe43.tar.gz
rspamd-1027e57ab831916be8bd5ae678e667ca93dffe43.zip
Rework test framework to use timeout instead of tests number
-rw-r--r--utils/rspamd_http_bench.c129
1 files changed, 81 insertions, 48 deletions
diff --git a/utils/rspamd_http_bench.c b/utils/rspamd_http_bench.c
index 12cbc6f25..42aa3a5bc 100644
--- a/utils/rspamd_http_bench.c
+++ b/utils/rspamd_http_bench.c
@@ -42,14 +42,15 @@ static gchar *server_key = NULL;
static guint cache_size = 10;
static guint nworkers = 1;
static gboolean openssl_mode = FALSE;
-double *latency, mean, std;
-static guint32 *pdiff;
static guint file_size = 500;
static guint pconns = 100;
-static guint ntests = 3000;
+static gdouble test_time = 10.0;
static rspamd_inet_addr_t *addr;
static gchar *latencies_file = NULL;
static guint32 workers_left = 0;
+static guint32 *conns_done = NULL;
+static const guint store_latencies = 100;
+static guint32 conns_pending = 0;
static GOptionEntry entries[] = {
{"port", 'p', 0, G_OPTION_ARG_INT, &port,
@@ -62,8 +63,8 @@ static GOptionEntry entries[] = {
"Size of payload to transfer (default: 500)", NULL},
{"conns", 'C', 0, G_OPTION_ARG_INT, &pconns,
"Number of parallel connections (default: 100)", NULL},
- {"tests", 't', 0, G_OPTION_ARG_INT, &ntests,
- "Number of tests to execute (default: 3000)", NULL},
+ {"time", 't', 0, G_OPTION_ARG_DOUBLE, &test_time,
+ "Time to run tests (default: 10.0 sec)", NULL},
{"openssl", 'o', 0, G_OPTION_ARG_NONE, &openssl_mode,
"Use openssl crypto", NULL},
{"host", 'h', 0, G_OPTION_ARG_STRING, &host,
@@ -75,6 +76,13 @@ static GOptionEntry entries[] = {
{NULL, 0, 0, G_OPTION_ARG_NONE, NULL, NULL, NULL}
};
+struct lat_elt {
+ gdouble lat;
+ guchar checked;
+};
+
+static struct lat_elt *latencies;
+
static gint
rspamd_client_body (struct rspamd_http_connection *conn,
struct rspamd_http_message *msg,
@@ -86,8 +94,10 @@ rspamd_client_body (struct rspamd_http_connection *conn,
}
struct client_cbdata {
- double *lat;
+ struct lat_elt *lat;
+ guint32 *wconns;
gdouble ts;
+ struct event_base *ev_base;
};
static void
@@ -107,16 +117,24 @@ rspamd_client_finish (struct rspamd_http_connection *conn,
{
struct client_cbdata *cb = conn->ud;
- *(cb->lat) = rspamd_get_ticks () - cb->ts;
+ cb->lat->lat = rspamd_get_ticks () - cb->ts;
+ cb->lat->checked = TRUE;
+ (*cb->wconns) ++;
+ conns_pending --;
close (conn->fd);
rspamd_http_connection_unref (conn);
g_free (cb);
+ if (conns_pending == 0) {
+ event_base_loopexit (cb->ev_base, NULL);
+ }
+
return 0;
}
static void
-rspamd_http_client_func (struct event_base *ev_base, double *latency,
+rspamd_http_client_func (struct event_base *ev_base, struct lat_elt *latency,
+ guint32 *wconns,
gpointer peer_key, gpointer client_key, struct rspamd_keypair_cache *c)
{
struct rspamd_http_message *msg;
@@ -146,20 +164,22 @@ rspamd_http_client_func (struct event_base *ev_base, double *latency,
cb = g_malloc (sizeof (*cb));
cb->ts = rspamd_get_ticks ();
cb->lat = latency;
+ cb->ev_base = ev_base;
+ cb->wconns = wconns;
+ latency->checked = FALSE;
rspamd_http_connection_write_message (conn, msg, NULL, NULL, cb,
fd, NULL, ev_base);
}
static void
-rspamd_worker_func (gpointer d)
+rspamd_worker_func (struct lat_elt *plat, guint32 *wconns)
{
guint i, j;
struct event_base *ev_base;
- gint *nt = d;
+ struct itimerval itv;
struct rspamd_keypair_cache *c = NULL;
gpointer client_key = NULL;
gpointer peer_key = NULL;
- gdouble ts1, ts2;
if (server_key) {
peer_key = rspamd_http_connection_make_peer_key (server_key);
@@ -170,49 +190,59 @@ rspamd_worker_func (gpointer d)
c = rspamd_keypair_cache_new (cache_size);
}
}
+
+ memset (&itv, 0, sizeof (itv));
+ double_to_tv (test_time, &itv.it_value);
+
ev_base = event_init ();
+ g_assert (setitimer (ITIMER_REAL, &itv, NULL) != -1);
- for (i = 0; i < ntests; i++) {
+ for (i = 0; ; i = (i + 1) % store_latencies) {
for (j = 0; j < pconns; j++) {
- rspamd_http_client_func (ev_base, &latency[(*nt) * pconns * ntests
- + i * pconns + j], peer_key, client_key, c);
+ rspamd_http_client_func (ev_base, &plat[i * pconns + j],
+ wconns, peer_key, client_key, c);
}
- ts1 = rspamd_get_ticks ();
- event_base_loop (ev_base, 0);
- ts2 = rspamd_get_ticks ();
+ conns_pending = pconns;
- g_atomic_int_add (pdiff, (guint32)((ts2 - ts1) * 1000000.));
+ event_base_loop (ev_base, 0);
}
}
static int
cmpd (const void *p1, const void *p2)
{
- const double *d1 = p1, *d2 = p2;
+ const struct lat_elt *d1 = p1, *d2 = p2;
- return (*d1) - (*d2);
+ return (d1->lat) - (d2->lat);
}
double
-rspamd_http_calculate_mean (double *lats, double *std)
+rspamd_http_calculate_mean (struct lat_elt *lats, double *std)
{
- guint i;
+ guint i, cnt, checked = 0;
gdouble mean = 0., dev = 0.;
- qsort (lats, ntests * pconns, sizeof (double), cmpd);
+ cnt = store_latencies * pconns;
+ qsort (lats, cnt, sizeof (*lats), cmpd);
- for (i = 0; i < ntests * pconns; i++) {
- mean += lats[i];
+ for (i = 0; i < cnt; i++) {
+ if (lats[i].checked) {
+ mean += lats[i].lat;
+ checked ++;
+ }
}
- mean /= ntests * pconns;
+ g_assert (checked > 0);
+ mean /= checked;
- for (i = 0; i < ntests * pconns; i++) {
- dev += (lats[i] - mean) * (lats[i] - mean);
+ for (i = 0; i < cnt; i++) {
+ if (lats[i].checked) {
+ dev += pow ((lats[i].lat - mean), 2);
+ }
}
- dev /= ntests * pconns;
+ dev /= checked;
*std = sqrt (dev);
return mean;
@@ -227,11 +257,9 @@ rspamd_http_start_workers (pid_t *sfd)
g_assert (sfd[i] != -1);
if (sfd[i] == 0) {
- gint *nt = g_malloc (sizeof (gint));
-
- *nt = i;
gperf_profiler_init (NULL, "http-bench");
- rspamd_worker_func (nt);
+ rspamd_worker_func (&latencies[i * pconns * store_latencies],
+ &conns_done[i]);
gperf_profiler_stop ();
exit (EXIT_SUCCESS);
}
@@ -283,8 +311,10 @@ main (int argc, char **argv)
struct event_base *ev_base;
rspamd_mempool_t *pool = rspamd_mempool_new (8192, "http-bench");
struct event term_ev, int_ev, cld_ev;
- gdouble total_diff;
+ gdouble total_done;
FILE *lat_file;
+ gdouble mean, std;
+ guint i;
rspamd_init_libs ();
@@ -311,11 +341,11 @@ main (int argc, char **argv)
g_assert (addr != NULL);
rspamd_inet_address_set_port (addr, port);
- latency = rspamd_mempool_alloc_shared (pool,
- nworkers * pconns * ntests * sizeof (gdouble));
+ latencies = rspamd_mempool_alloc_shared (pool,
+ nworkers * pconns * store_latencies * sizeof (*latencies));
sfd = g_malloc (sizeof (*sfd) * nworkers);
- pdiff = rspamd_mempool_alloc_shared (pool, sizeof (guint32));
- *pdiff = 0;
+ conns_done = rspamd_mempool_alloc_shared (pool, sizeof (guint32) * nworkers);
+ memset (conns_done, 0, sizeof (guint32) * nworkers);
rspamd_http_start_workers (sfd);
@@ -334,15 +364,18 @@ main (int argc, char **argv)
event_base_loop (ev_base, 0);
- total_diff = *pdiff / nworkers / 1000000.0;
+ total_done = 0;
+ for (i = 0; i < nworkers; i ++) {
+ total_done += conns_done[i];
+ }
rspamd_printf ("Made %d connections of size %d in %.6fs, %.6f cps, %.6f MB/sec\n",
- nworkers * ntests * pconns,
+ (gint)total_done,
file_size,
- total_diff,
- nworkers * ntests * pconns / total_diff,
- nworkers * ntests * pconns * file_size / total_diff / (1024.0 * 1024.0));
- mean = rspamd_http_calculate_mean (latency, &std);
+ test_time,
+ total_done / test_time,
+ total_done * file_size / test_time / (1024.0 * 1024.0));
+ mean = rspamd_http_calculate_mean (latencies, &std);
rspamd_printf ("Latency: %.6f ms mean, %.6f dev\n",
mean * 1000.0, std * 1000.0);
@@ -350,10 +383,10 @@ main (int argc, char **argv)
lat_file = fopen (latencies_file, "w");
if (lat_file) {
- guint i;
-
- for (i = 0; i < nworkers * pconns * ntests; i ++) {
- rspamd_fprintf (lat_file, "%.6f\n", latency[i]);
+ for (i = 0; i < store_latencies * pconns; i ++) {
+ if (latencies[i].checked) {
+ rspamd_fprintf (lat_file, "%.6f\n", latencies[i].lat);
+ }
}
fclose (lat_file);