]>
git.saurik.com Git - redis.git/blob - benchmark.c 
   1  /* Redis benchmark utility.    3   * Copyright (c) 2006-2009, 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.   49  #define REPLY_RETCODE 1   53  #define CLIENT_CONNECTING 0   54  #define CLIENT_SENDQUERY 1   55  #define CLIENT_READREPLY 2   57  #define MAX_LATENCY 5000   59  #define REDIS_NOTUSED(V) ((void) V)   61  static struct  config 
{   69      int  randomkeys_keyspacelen
;   82  typedef  struct  _client 
{   87      int  mbulk
;           /* Number of elements in an mbulk reply */   88      int  readlen
;         /* readlen == -1 means read a single line */   89      unsigned int  written
;         /* bytes of 'obuf' already written */   91      long long  start
;     /* start time in milliseconds */   95  static void  writeHandler ( aeEventLoop 
* el
,  int  fd
,  void  * privdata
,  int  mask
);   96  static void  createMissingClients ( client c
);   99  static long long  mstime ( void ) {  103      gettimeofday (& tv
,  NULL
);  104      mst 
= (( long ) tv
. tv_sec
)* 1000 ;  105      mst 
+=  tv
. tv_usec
/ 1000 ;  109  static void  freeClient ( client c
) {  112      aeDeleteFileEvent ( config
. el
, c
-> fd
, AE_WRITABLE
);  113      aeDeleteFileEvent ( config
. el
, c
-> fd
, AE_READABLE
);  118      config
. liveclients
--;  119      ln 
=  listSearchKey ( config
. clients
, c
);  121      listDelNode ( config
. clients
, ln
);  124  static void  freeAllClients ( void ) {  125      listNode 
* ln 
=  config
. clients
-> head
, * next
;  129          freeClient ( ln
-> value
);  134  static void  resetClient ( client c
) {  135      aeDeleteFileEvent ( config
. el
, c
-> fd
, AE_WRITABLE
);  136      aeDeleteFileEvent ( config
. el
, c
-> fd
, AE_READABLE
);  137      aeCreateFileEvent ( config
. el
, c
-> fd
,  AE_WRITABLE
, writeHandler
, c
, NULL
);  139      c
-> ibuf 
=  sdsempty ();  140      c
-> readlen 
= ( c
-> replytype 
==  REPLY_BULK
) ? - 1  :  0 ;  143      c
-> state 
=  CLIENT_SENDQUERY
;  145      createMissingClients ( c
);  148  static void  randomizeClientKey ( client c
) {  153      p 
=  strstr ( c
-> obuf
,  "_rand" );  156      r 
=  random () %  config
. randomkeys_keyspacelen
;  157      sprintf ( buf
, " %l d" , r
);  158      memcpy ( p
, buf
, strlen ( buf
));  161  static void  prepareClientForReply ( client c
,  int  type
) {  162      if  ( type 
==  REPLY_BULK
) {  163          c
-> replytype 
=  REPLY_BULK
;  165      }  else if  ( type 
==  REPLY_MBULK
) {  166          c
-> replytype 
=  REPLY_MBULK
;  175  static void  clientDone ( client c
) {  177      config
. donerequests 
++;  178      latency 
=  mstime () -  c
-> start
;  179      if  ( latency 
>  MAX_LATENCY
)  latency 
=  MAX_LATENCY
;  180      config
. latency
[ latency
]++;  182      if  ( config
. donerequests 
==  config
. requests
) {  187      if  ( config
. keepalive
) {  189          if  ( config
. randomkeys
)  randomizeClientKey ( c
);  191          config
. liveclients
--;  192          createMissingClients ( c
);  193          config
. liveclients
++;  198  static void  readHandler ( aeEventLoop 
* el
,  int  fd
,  void  * privdata
,  int  mask
)  207      nread 
=  read ( c
-> fd
,  buf
,  1024 );  209          fprintf ( stderr
,  "Reading from socket:  %s \n " ,  strerror ( errno
));  214          fprintf ( stderr
,  "EOF from client \n " );  218      c
-> ibuf 
=  sdscatlen ( c
-> ibuf
, buf
, nread
);  221      if  ( c
-> replytype 
==  REPLY_INT 
||  222          c
-> replytype 
==  REPLY_RETCODE 
||  223          ( c
-> replytype 
==  REPLY_BULK 
&&  c
-> readlen 
== - 1 ) ||  224          ( c
-> replytype 
==  REPLY_MBULK 
&&  c
-> readlen 
== - 1 ) ||  225          ( c
-> replytype 
==  REPLY_MBULK 
&&  c
-> mbulk 
== - 1 )) {  228          if  (( p 
=  strchr ( c
-> ibuf
, ' \n ' )) !=  NULL
) {  229              if  ( c
-> replytype 
==  REPLY_BULK 
||  230                  ( c
-> replytype 
==  REPLY_MBULK 
&&  c
-> mbulk 
!= - 1 ))  234                  c
-> readlen 
=  atoi ( c
-> ibuf
+ 1 )+ 2 ;  235                  /* Handle null bulk reply "$-1" */  236                  if  ( c
-> readlen
- 2  == - 1 ) {  240                  /* Leave all the rest in the input buffer */  241                  c
-> ibuf 
=  sdsrange ( c
-> ibuf
,( p
- c
-> ibuf
)+ 1 ,- 1 );  242              }  else if  ( c
-> replytype 
==  REPLY_MBULK 
&&  c
-> mbulk 
== - 1 ) {  245                  c
-> mbulk 
=  atoi ( c
-> ibuf
+ 1 );  246                  /* Handle null bulk reply "*-1" */  247                  if  ( c
-> mbulk 
== - 1 ) {  251                  /* Leave all the rest in the input buffer */  252                  c
-> ibuf 
=  sdsrange ( c
-> ibuf
,( p
- c
-> ibuf
)+ 1 ,- 1 );  255                  c
-> ibuf 
=  sdstrim ( c
-> ibuf
, " \r\n " );  261      /* bulk read, did we read everything? */  262      if  (( c
-> replytype 
==  REPLY_BULK 
||  c
-> replytype 
==  REPLY_MBULK
) &&  263          ( unsigned ) c
-> readlen 
<=  sdslen ( c
-> ibuf
))  265          if  ( c
-> replytype 
==  REPLY_BULK
) {  267          }  else if  ( c
-> replytype 
==  REPLY_MBULK
) {  268              if  (-- c
-> mbulk 
==  0 ) {  271                  c
-> ibuf 
=  sdsrange ( c
-> ibuf
, c
-> readlen
,- 1 );  279  static void  writeHandler ( aeEventLoop 
* el
,  int  fd
,  void  * privdata
,  int  mask
)  286      if  ( c
-> state 
==  CLIENT_CONNECTING
) {  287          c
-> state 
=  CLIENT_SENDQUERY
;  290      if  ( sdslen ( c
-> obuf
) >  c
-> written
) {  291          void  * ptr 
=  c
-> obuf
+ c
-> written
;  292          int  len 
=  sdslen ( c
-> obuf
) -  c
-> written
;  293          int  nwritten 
=  write ( c
-> fd
,  ptr
,  len
);  294          if  ( nwritten 
== - 1 ) {  295              fprintf ( stderr
,  "Writing to socket:  %s \n " ,  strerror ( errno
));  299          c
-> written 
+=  nwritten
;  300          if  ( sdslen ( c
-> obuf
) ==  c
-> written
) {  301              aeDeleteFileEvent ( config
. el
, c
-> fd
, AE_WRITABLE
);  302              aeCreateFileEvent ( config
. el
, c
-> fd
, AE_READABLE
, readHandler
, c
, NULL
);  303              c
-> state 
=  CLIENT_READREPLY
;  308  static  client 
createClient ( void ) {  309      client c 
=  zmalloc ( sizeof ( struct  _client
));  310      char  err
[ ANET_ERR_LEN
];  312      c
-> fd 
=  anetTcpNonBlockConnect ( err
, config
. hostip
, config
. hostport
);  313      if  ( c
-> fd 
==  ANET_ERR
) {  315          fprintf ( stderr
, "Connect:  %s \n " , err
);  318      anetTcpNoDelay ( NULL
, c
-> fd
);  319      c
-> obuf 
=  sdsempty ();  320      c
-> ibuf 
=  sdsempty ();  324      c
-> state 
=  CLIENT_CONNECTING
;  325      aeCreateFileEvent ( config
. el
,  c
-> fd
,  AE_WRITABLE
,  writeHandler
,  c
,  NULL
);  326      config
. liveclients
++;  327      listAddNodeTail ( config
. clients
, c
);  331  static void  createMissingClients ( client c
) {  332      while ( config
. liveclients 
<  config
. numclients
) {  333          client 
new  =  createClient ();  336          new -> obuf 
=  sdsdup ( c
-> obuf
);  337          if  ( config
. randomkeys
)  randomizeClientKey ( c
);  338          new -> replytype 
=  c
-> replytype
;  339          if  ( c
-> replytype 
==  REPLY_BULK
)  344  static void  showLatencyReport ( char  * title
) {  346      float  perc
,  reqpersec
;  348      reqpersec 
= ( float ) config
. donerequests
/(( float ) config
. totlatency
/ 1000 );  350          printf ( "======  %s  ====== \n " ,  title
);  351          printf ( "   %d  requests completed in %.2f seconds \n " ,  config
. donerequests
,  352              ( float ) config
. totlatency
/ 1000 );  353          printf ( "   %d  parallel clients \n " ,  config
. numclients
);  354          printf ( "   %d  bytes payload \n " ,  config
. datasize
);  355          printf ( "  keep alive:  %d \n " ,  config
. keepalive
);  357          for  ( j 
=  0 ;  j 
<=  MAX_LATENCY
;  j
++) {  358              if  ( config
. latency
[ j
]) {  359                  seen 
+=  config
. latency
[ j
];  360                  perc 
= (( float ) seen
* 100 )/ config
. donerequests
;  361                  printf ( "%.2f%% <=  %d  milliseconds \n " ,  perc
,  j
);  364          printf ( "%.2f requests per second \n\n " ,  reqpersec
);  366          printf ( " %s : %.2f requests per second \n " ,  title
,  reqpersec
);  370  static void  prepareForBenchmark ( void )  372      memset ( config
. latency
, 0 , sizeof ( int )*( MAX_LATENCY
+ 1 ));  373      config
. start 
=  mstime ();  374      config
. donerequests 
=  0 ;  377  static void  endBenchmark ( char  * title
) {  378      config
. totlatency 
=  mstime ()- config
. start
;  379      showLatencyReport ( title
);  383  void  parseOptions ( int  argc
,  char  ** argv
) {  386      for  ( i 
=  1 ;  i 
<  argc
;  i
++) {  387          int  lastarg 
=  i
== argc
- 1 ;  389          if  (! strcmp ( argv
[ i
], "-c" ) && ! lastarg
) {  390              config
. numclients 
=  atoi ( argv
[ i
+ 1 ]);  392          }  else if  (! strcmp ( argv
[ i
], "-n" ) && ! lastarg
) {  393              config
. requests 
=  atoi ( argv
[ i
+ 1 ]);  395          }  else if  (! strcmp ( argv
[ i
], "-k" ) && ! lastarg
) {  396              config
. keepalive 
=  atoi ( argv
[ i
+ 1 ]);  398          }  else if  (! strcmp ( argv
[ i
], "-h" ) && ! lastarg
) {  399              char  * ip 
=  zmalloc ( 32 );  400              if  ( anetResolve ( NULL
, argv
[ i
+ 1 ], ip
) ==  ANET_ERR
) {  401                  printf ( "Can't resolve  %s \n " ,  argv
[ i
]);  406          }  else if  (! strcmp ( argv
[ i
], "-p" ) && ! lastarg
) {  407              config
. hostport 
=  atoi ( argv
[ i
+ 1 ]);  409          }  else if  (! strcmp ( argv
[ i
], "-d" ) && ! lastarg
) {  410              config
. datasize 
=  atoi ( argv
[ i
+ 1 ]);  412              if  ( config
. datasize 
<  1 )  config
. datasize
= 1 ;  413              if  ( config
. datasize 
>  1024 * 1024 )  config
. datasize 
=  1024 * 1024 ;  414          }  else if  (! strcmp ( argv
[ i
], "-r" ) && ! lastarg
) {  415              config
. randomkeys 
=  1 ;  416              config
. randomkeys_keyspacelen 
=  atoi ( argv
[ i
+ 1 ]);  417              if  ( config
. randomkeys_keyspacelen 
<  0 )  418                  config
. randomkeys_keyspacelen 
=  0 ;  420          }  else if  (! strcmp ( argv
[ i
], "-q" )) {  422          }  else if  (! strcmp ( argv
[ i
], "-l" )) {  425              printf ( "Wrong option ' %s ' or option argument missing \n\n " , argv
[ i
]);  426              printf ( "Usage: redis-benchmark [-h <host>] [-p <port>] [-c <clients>] [-n <requests]> [-k <boolean>] \n\n " );  427              printf ( " -h <hostname>      Server hostname (default 127.0.0.1) \n " );  428              printf ( " -p <hostname>      Server port (default 6379) \n " );  429              printf ( " -c <clients>       Number of parallel connections (default 50) \n " );  430              printf ( " -n <requests>      Total number of requests (default 10000) \n " );  431              printf ( " -d <size>          Data size of SET/GET value in bytes (default 2) \n " );  432              printf ( " -k <boolean>       1=keep alive 0=reconnect (default 1) \n " );  433              printf ( " -r <keyspacelen>   Use random keys for SET/GET/INCR \n " );  434              printf ( "  Using this option the benchmark will get/set keys \n " );  435              printf ( "  in the form mykey_rand000000012456 instead of constant \n " );  436              printf ( "  keys, the <keyspacelen> argument determines the max \n " );  437              printf ( "  number of values for the random number. For instance \n " );  438              printf ( "  if set to 10 only rand000000000000 - rand000000000009 \n " );  439              printf ( "  range will be allowed. \n " );  440              printf ( " -q                 Quiet. Just show query/sec values \n " );  441              printf ( " -l                 Loop. Run the tests forever \n " );  447  int  main ( int  argc
,  char  ** argv
) {  450      signal ( SIGHUP
,  SIG_IGN
);  451      signal ( SIGPIPE
,  SIG_IGN
);  453      config
. numclients 
=  50 ;  454      config
. requests 
=  10000 ;  455      config
. liveclients 
=  0 ;  456      config
. el 
=  aeCreateEventLoop ();  457      config
. keepalive 
=  1 ;  458      config
. donerequests 
=  0 ;  460      config
. randomkeys 
=  0 ;  461      config
. randomkeys_keyspacelen 
=  0 ;  464      config
. latency 
=  NULL
;  465      config
. clients 
=  listCreate ();  466      config
. latency 
=  zmalloc ( sizeof ( int )*( MAX_LATENCY
+ 1 ));  468      config
. hostip 
=  "127.0.0.1" ;  469      config
. hostport 
=  6379 ;  471      parseOptions ( argc
, argv
);  473      if  ( config
. keepalive 
==  0 ) {  474          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 " );  478          prepareForBenchmark ();  481          c
-> obuf 
=  sdscatprintf ( c
-> obuf
, "SET foo_rand000000000000  %d \r\n " , config
. datasize
);  483              char  * data 
=  zmalloc ( config
. datasize
+ 2 );  484              memset ( data
, 'x' , config
. datasize
);  485              data
[ config
. datasize
] =  ' \r ' ;  486              data
[ config
. datasize
+ 1 ] =  ' \n ' ;  487              c
-> obuf 
=  sdscatlen ( c
-> obuf
, data
, config
. datasize
+ 2 );  489          prepareClientForReply ( c
, REPLY_RETCODE
);  490          createMissingClients ( c
);  494          prepareForBenchmark ();  497          c
-> obuf 
=  sdscat ( c
-> obuf
, "GET foo_rand000000000000 \r\n " );  498          prepareClientForReply ( c
, REPLY_BULK
);  499          createMissingClients ( c
);  503          prepareForBenchmark ();  506          c
-> obuf 
=  sdscat ( c
-> obuf
, "INCR counter_rand000000000000 \r\n " );  507          prepareClientForReply ( c
, REPLY_INT
);  508          createMissingClients ( c
);  510          endBenchmark ( "INCR" );  512          prepareForBenchmark ();  515          c
-> obuf 
=  sdscat ( c
-> obuf
, "LPUSH mylist 3 \r\n bar \r\n " );  516          prepareClientForReply ( c
, REPLY_INT
);  517          createMissingClients ( c
);  519          endBenchmark ( "LPUSH" );  521          prepareForBenchmark ();  524          c
-> obuf 
=  sdscat ( c
-> obuf
, "LPOP mylist \r\n " );  525          prepareClientForReply ( c
, REPLY_BULK
);  526          createMissingClients ( c
);  528          endBenchmark ( "LPOP" );  530          prepareForBenchmark ();  533          c
-> obuf 
=  sdscat ( c
-> obuf
, "PING \r\n " );  534          prepareClientForReply ( c
, REPLY_RETCODE
);  535          createMissingClients ( c
);  537          endBenchmark ( "PING" );  539          prepareForBenchmark ();  542          c
-> obuf 
=  sdscat ( c
-> obuf
, "LPUSH mylist 3 \r\n bar \r\n " );  543          prepareClientForReply ( c
, REPLY_RETCODE
);  544          createMissingClients ( c
);  546          endBenchmark ( "LPUSH (again, in order to bench LRANGE)" );  548          prepareForBenchmark ();  551          c
-> obuf 
=  sdscat ( c
-> obuf
, "LRANGE mylist 0 99 \r\n " );  552          prepareClientForReply ( c
, REPLY_MBULK
);  553          createMissingClients ( c
);  555          endBenchmark ( "LRANGE (first 100 elements)" );  558      }  while ( config
. loop
);