1 /* Redis benchmark utility.
3 * Copyright (c) 2009-2010, Salvatore Sanfilippo <antirez at gmail dot com>
6 * Redistribution and use in source and binary forms, with or without
7 * modification, are permitted provided that the following conditions are met:
9 * * Redistributions of source code must retain the above copyright notice,
10 * this list of conditions and the following disclaimer.
11 * * Redistributions in binary form must reproduce the above copyright
12 * notice, this list of conditions and the following disclaimer in the
13 * documentation and/or other materials provided with the distribution.
14 * * Neither the name of Redis nor the names of its contributors may be used
15 * to endorse or promote products derived from this software without
16 * specific prior written permission.
18 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
19 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
20 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
21 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
22 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
23 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
24 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
25 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
26 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
27 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
28 * POSSIBILITY OF SUCH DAMAGE.
48 #define CLIENT_CONNECTING 0
49 #define CLIENT_SENDQUERY 1
50 #define CLIENT_READREPLY 2
52 #define REDIS_NOTUSED(V) ((void) V)
54 static struct config
{
63 int randomkeys_keyspacelen
;
79 typedef struct _client
{
80 redisContext
*context
;
83 unsigned int written
; /* bytes of 'obuf' already written */
85 long long start
; /* start time of a request */
86 long long latency
; /* request latency */
90 static void writeHandler(aeEventLoop
*el
, int fd
, void *privdata
, int mask
);
91 static void createMissingClients(client c
);
94 static long long ustime(void) {
98 gettimeofday(&tv
, NULL
);
99 ust
= ((long)tv
.tv_sec
)*1000000;
104 static long long mstime(void) {
108 gettimeofday(&tv
, NULL
);
109 mst
= ((long)tv
.tv_sec
)*1000;
110 mst
+= tv
.tv_usec
/1000;
114 static void freeClient(client c
) {
116 aeDeleteFileEvent(config
.el
,c
->context
->fd
,AE_WRITABLE
);
117 aeDeleteFileEvent(config
.el
,c
->context
->fd
,AE_READABLE
);
118 redisFree(c
->context
);
121 config
.liveclients
--;
122 ln
= listSearchKey(config
.clients
,c
);
124 listDelNode(config
.clients
,ln
);
127 static void freeAllClients(void) {
128 listNode
*ln
= config
.clients
->head
, *next
;
132 freeClient(ln
->value
);
137 static void resetClient(client c
) {
138 aeDeleteFileEvent(config
.el
,c
->context
->fd
,AE_WRITABLE
);
139 aeDeleteFileEvent(config
.el
,c
->context
->fd
,AE_READABLE
);
140 aeCreateFileEvent(config
.el
,c
->context
->fd
,AE_WRITABLE
,writeHandler
,c
);
142 c
->state
= CLIENT_SENDQUERY
;
147 static void randomizeClientKey(client c
) {
152 p
= strstr(c
->obuf
, "_rand");
155 r
= random() % config
.randomkeys_keyspacelen
;
156 sprintf(buf
,"%ld",r
);
157 memcpy(p
,buf
,strlen(buf
));
160 static void clientDone(client c
) {
161 if (config
.donerequests
== config
.requests
) {
166 if (config
.keepalive
) {
168 if (config
.randomkeys
) randomizeClientKey(c
);
170 config
.liveclients
--;
171 createMissingClients(c
);
172 config
.liveclients
++;
177 static void readHandler(aeEventLoop
*el
, int fd
, void *privdata
, int mask
) {
184 /* Calculate latency only for the first read event. This means that the
185 * server already sent the reply and we need to parse it. Parsing overhead
186 * is not part of the latency, so calculate it only once, here. */
187 if (c
->latency
< 0) c
->latency
= ustime()-(c
->start
);
189 if (redisBufferRead(c
->context
) != REDIS_OK
) {
190 fprintf(stderr
,"Error: %s\n",c
->context
->errstr
);
193 if (redisGetReply(c
->context
,&reply
) != REDIS_OK
) {
194 fprintf(stderr
,"Error: %s\n",c
->context
->errstr
);
198 if (config
.donerequests
< config
.requests
)
199 config
.latency
[config
.donerequests
++] = c
->latency
;
205 static void writeHandler(aeEventLoop
*el
, int fd
, void *privdata
, int mask
) {
211 if (c
->state
== CLIENT_CONNECTING
) {
212 c
->state
= CLIENT_SENDQUERY
;
216 if (sdslen(c
->obuf
) > c
->written
) {
217 void *ptr
= c
->obuf
+c
->written
;
218 int nwritten
= write(c
->context
->fd
,ptr
,sdslen(c
->obuf
)-c
->written
);
219 if (nwritten
== -1) {
221 fprintf(stderr
, "Writing to socket: %s\n", strerror(errno
));
225 c
->written
+= nwritten
;
226 if (sdslen(c
->obuf
) == c
->written
) {
227 aeDeleteFileEvent(config
.el
,c
->context
->fd
,AE_WRITABLE
);
228 aeCreateFileEvent(config
.el
,c
->context
->fd
,AE_READABLE
,readHandler
,c
);
229 c
->state
= CLIENT_READREPLY
;
234 static client
createClient(int replytype
) {
235 client c
= zmalloc(sizeof(struct _client
));
236 if (config
.hostsocket
== NULL
) {
237 c
->context
= redisConnectNonBlock(config
.hostip
,config
.hostport
);
239 c
->context
= redisConnectUnixNonBlock(config
.hostsocket
);
241 if (c
->context
->err
) {
242 fprintf(stderr
,"Could not connect to Redis at ");
243 if (config
.hostsocket
== NULL
)
244 fprintf(stderr
,"%s:%d: %s\n",config
.hostip
,config
.hostport
,c
->context
->errstr
);
246 fprintf(stderr
,"%s: %s\n",config
.hostsocket
,c
->context
->errstr
);
249 c
->replytype
= replytype
;
250 c
->state
= CLIENT_CONNECTING
;
251 c
->obuf
= sdsempty();
253 redisSetReplyObjectFunctions(c
->context
,NULL
);
254 aeCreateFileEvent(config
.el
,c
->context
->fd
,AE_WRITABLE
,writeHandler
,c
);
255 listAddNodeTail(config
.clients
,c
);
256 config
.liveclients
++;
260 static void createMissingClients(client c
) {
261 while(config
.liveclients
< config
.numclients
) {
262 client
new = createClient(c
->replytype
);
264 new->obuf
= sdsdup(c
->obuf
);
265 if (config
.randomkeys
) randomizeClientKey(c
);
269 static int compareLatency(const void *a
, const void *b
) {
270 return (*(long long*)a
)-(*(long long*)b
);
273 static void showLatencyReport(void) {
275 float perc
, reqpersec
;
277 reqpersec
= (float)config
.donerequests
/((float)config
.totlatency
/1000);
279 printf("====== %s ======\n", config
.title
);
280 printf(" %d requests completed in %.2f seconds\n", config
.donerequests
,
281 (float)config
.totlatency
/1000);
282 printf(" %d parallel clients\n", config
.numclients
);
283 printf(" %d bytes payload\n", config
.datasize
);
284 printf(" keep alive: %d\n", config
.keepalive
);
287 qsort(config
.latency
,config
.requests
,sizeof(long long),compareLatency
);
288 for (i
= 0; i
< config
.requests
; i
++) {
289 if (config
.latency
[i
]/1000 != curlat
|| i
== (config
.requests
-1)) {
290 curlat
= config
.latency
[i
]/1000;
291 perc
= ((float)(i
+1)*100)/config
.requests
;
292 printf("%.2f%% <= %d milliseconds\n", perc
, curlat
);
295 printf("%.2f requests per second\n\n", reqpersec
);
297 printf("%s: %.2f requests per second\n", config
.title
, reqpersec
);
301 static void prepareForBenchmark(char *title
) {
302 config
.title
= title
;
303 config
.start
= mstime();
304 config
.donerequests
= 0;
307 static void endBenchmark(void) {
308 config
.totlatency
= mstime()-config
.start
;
313 void parseOptions(int argc
, char **argv
) {
316 for (i
= 1; i
< argc
; i
++) {
317 int lastarg
= i
==argc
-1;
319 if (!strcmp(argv
[i
],"-c") && !lastarg
) {
320 config
.numclients
= atoi(argv
[i
+1]);
322 } else if (!strcmp(argv
[i
],"-n") && !lastarg
) {
323 config
.requests
= atoi(argv
[i
+1]);
325 } else if (!strcmp(argv
[i
],"-k") && !lastarg
) {
326 config
.keepalive
= atoi(argv
[i
+1]);
328 } else if (!strcmp(argv
[i
],"-h") && !lastarg
) {
329 config
.hostip
= argv
[i
+1];
331 } else if (!strcmp(argv
[i
],"-p") && !lastarg
) {
332 config
.hostport
= atoi(argv
[i
+1]);
334 } else if (!strcmp(argv
[i
],"-s") && !lastarg
) {
335 config
.hostsocket
= argv
[i
+1];
337 } else if (!strcmp(argv
[i
],"-d") && !lastarg
) {
338 config
.datasize
= atoi(argv
[i
+1]);
340 if (config
.datasize
< 1) config
.datasize
=1;
341 if (config
.datasize
> 1024*1024) config
.datasize
= 1024*1024;
342 } else if (!strcmp(argv
[i
],"-r") && !lastarg
) {
343 config
.randomkeys
= 1;
344 config
.randomkeys_keyspacelen
= atoi(argv
[i
+1]);
345 if (config
.randomkeys_keyspacelen
< 0)
346 config
.randomkeys_keyspacelen
= 0;
348 } else if (!strcmp(argv
[i
],"-q")) {
350 } else if (!strcmp(argv
[i
],"-l")) {
352 } else if (!strcmp(argv
[i
],"-D")) {
354 } else if (!strcmp(argv
[i
],"-I")) {
357 printf("Wrong option '%s' or option argument missing\n\n",argv
[i
]);
358 printf("Usage: redis-benchmark [-h <host>] [-p <port>] [-c <clients>] [-n <requests]> [-k <boolean>]\n\n");
359 printf(" -h <hostname> Server hostname (default 127.0.0.1)\n");
360 printf(" -p <port> Server port (default 6379)\n");
361 printf(" -s <socket> Server socket (overrides host and port)\n");
362 printf(" -c <clients> Number of parallel connections (default 50)\n");
363 printf(" -n <requests> Total number of requests (default 10000)\n");
364 printf(" -d <size> Data size of SET/GET value in bytes (default 2)\n");
365 printf(" -k <boolean> 1=keep alive 0=reconnect (default 1)\n");
366 printf(" -r <keyspacelen> Use random keys for SET/GET/INCR, random values for SADD\n");
367 printf(" Using this option the benchmark will get/set keys\n");
368 printf(" in the form mykey_rand000000012456 instead of constant\n");
369 printf(" keys, the <keyspacelen> argument determines the max\n");
370 printf(" number of values for the random number. For instance\n");
371 printf(" if set to 10 only rand000000000000 - rand000000000009\n");
372 printf(" range will be allowed.\n");
373 printf(" -q Quiet. Just show query/sec values\n");
374 printf(" -l Loop. Run the tests forever\n");
375 printf(" -I Idle mode. Just open N idle connections and wait.\n");
376 printf(" -D Debug mode. more verbose.\n");
382 int showThroughput(struct aeEventLoop
*eventLoop
, long long id
, void *clientData
) {
383 REDIS_NOTUSED(eventLoop
);
385 REDIS_NOTUSED(clientData
);
387 float dt
= (float)(mstime()-config
.start
)/1000.0;
388 float rps
= (float)config
.donerequests
/dt
;
389 printf("%s: %.2f\r", config
.title
, rps
);
391 return 250; /* every 250ms */
394 int main(int argc
, char **argv
) {
397 signal(SIGHUP
, SIG_IGN
);
398 signal(SIGPIPE
, SIG_IGN
);
401 config
.numclients
= 50;
402 config
.requests
= 10000;
403 config
.liveclients
= 0;
404 config
.el
= aeCreateEventLoop();
405 aeCreateTimeEvent(config
.el
,1,showThroughput
,NULL
,NULL
);
406 config
.keepalive
= 1;
407 config
.donerequests
= 0;
409 config
.randomkeys
= 0;
410 config
.randomkeys_keyspacelen
= 0;
414 config
.latency
= NULL
;
415 config
.clients
= listCreate();
416 config
.hostip
= "127.0.0.1";
417 config
.hostport
= 6379;
418 config
.hostsocket
= NULL
;
420 parseOptions(argc
,argv
);
421 config
.latency
= zmalloc(sizeof(long long)*config
.requests
);
423 if (config
.keepalive
== 0) {
424 printf("WARNING: keepalive disabled, you probably need 'echo 1 > /proc/sys/net/ipv4/tcp_tw_reuse' for Linux and 'sudo sysctl -w net.inet.tcp.msl=1000' for Mac OS X in order to use a lot of clients/requests\n");
427 if (config
.idlemode
) {
428 printf("Creating %d idle connections and waiting forever (Ctrl+C when done)\n", config
.numclients
);
429 prepareForBenchmark("IDLE");
430 c
= createClient(0); /* will never receive a reply */
431 c
->obuf
= sdsempty();
432 createMissingClients(c
);
434 /* and will wait for every */
438 prepareForBenchmark("PING");
439 c
= createClient(REDIS_REPLY_STATUS
);
440 c
->obuf
= sdscat(c
->obuf
,"PING\r\n");
441 createMissingClients(c
);
445 prepareForBenchmark("PING (multi bulk)");
446 c
= createClient(REDIS_REPLY_STATUS
);
447 c
->obuf
= sdscat(c
->obuf
,"*1\r\n$4\r\nPING\r\n");
448 createMissingClients(c
);
452 prepareForBenchmark("MSET (10 keys, multi bulk)");
453 c
= createClient(REDIS_REPLY_ARRAY
);
454 c
->obuf
= sdscatprintf(c
->obuf
,"*%d\r\n$4\r\nMSET\r\n", 11);
457 char *data
= zmalloc(config
.datasize
+2);
458 memset(data
,'x',config
.datasize
);
459 for (i
= 0; i
< 10; i
++) {
460 c
->obuf
= sdscatprintf(c
->obuf
,"$%d\r\n%s\r\n",config
.datasize
,data
);
464 createMissingClients(c
);
468 prepareForBenchmark("SET");
469 c
= createClient(REDIS_REPLY_STATUS
);
470 c
->obuf
= sdscat(c
->obuf
,"*3\r\n$3\r\nSET\r\n$20\r\nfoo_rand000000000000\r\n");
472 char *data
= zmalloc(config
.datasize
+2);
473 memset(data
,'x',config
.datasize
);
474 data
[config
.datasize
] = '\r';
475 data
[config
.datasize
+1] = '\n';
476 c
->obuf
= sdscatprintf(c
->obuf
,"$%d\r\n",config
.datasize
);
477 c
->obuf
= sdscatlen(c
->obuf
,data
,config
.datasize
+2);
479 createMissingClients(c
);
483 prepareForBenchmark("GET");
484 c
= createClient(REDIS_REPLY_STRING
);
485 c
->obuf
= sdscat(c
->obuf
,"GET foo_rand000000000000\r\n");
486 createMissingClients(c
);
490 prepareForBenchmark("INCR");
491 c
= createClient(REDIS_REPLY_INTEGER
);
492 c
->obuf
= sdscat(c
->obuf
,"INCR counter_rand000000000000\r\n");
493 createMissingClients(c
);
497 prepareForBenchmark("LPUSH");
498 c
= createClient(REDIS_REPLY_INTEGER
);
499 c
->obuf
= sdscat(c
->obuf
,"LPUSH mylist bar\r\n");
500 createMissingClients(c
);
504 prepareForBenchmark("LPOP");
505 c
= createClient(REDIS_REPLY_STRING
);
506 c
->obuf
= sdscat(c
->obuf
,"LPOP mylist\r\n");
507 createMissingClients(c
);
511 prepareForBenchmark("SADD");
512 c
= createClient(REDIS_REPLY_STATUS
);
513 c
->obuf
= sdscat(c
->obuf
,"SADD myset counter_rand000000000000\r\n");
514 createMissingClients(c
);
518 prepareForBenchmark("SPOP");
519 c
= createClient(REDIS_REPLY_STRING
);
520 c
->obuf
= sdscat(c
->obuf
,"SPOP myset\r\n");
521 createMissingClients(c
);
525 prepareForBenchmark("LPUSH (again, in order to bench LRANGE)");
526 c
= createClient(REDIS_REPLY_STATUS
);
527 c
->obuf
= sdscat(c
->obuf
,"LPUSH mylist bar\r\n");
528 createMissingClients(c
);
532 prepareForBenchmark("LRANGE (first 100 elements)");
533 c
= createClient(REDIS_REPLY_ARRAY
);
534 c
->obuf
= sdscat(c
->obuf
,"LRANGE mylist 0 99\r\n");
535 createMissingClients(c
);
539 prepareForBenchmark("LRANGE (first 300 elements)");
540 c
= createClient(REDIS_REPLY_ARRAY
);
541 c
->obuf
= sdscat(c
->obuf
,"LRANGE mylist 0 299\r\n");
542 createMissingClients(c
);
546 prepareForBenchmark("LRANGE (first 450 elements)");
547 c
= createClient(REDIS_REPLY_ARRAY
);
548 c
->obuf
= sdscat(c
->obuf
,"LRANGE mylist 0 449\r\n");
549 createMissingClients(c
);
553 prepareForBenchmark("LRANGE (first 600 elements)");
554 c
= createClient(REDIS_REPLY_ARRAY
);
555 c
->obuf
= sdscat(c
->obuf
,"LRANGE mylist 0 599\r\n");
556 createMissingClients(c
);
561 } while(config
.loop
);