]>
git.saurik.com Git - redis.git/blob - redis-cli.c
cfd2081ac1a387781542d85d4cfaef7326943f05
   1 /* Redis CLI (command line interface) 
   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. 
  43 #define REDIS_CMD_INLINE 1 
  44 #define REDIS_CMD_BULK 2 
  45 #define REDIS_CMD_MULTIBULK 3 
  47 #define REDIS_NOTUSED(V) ((void) V) 
  49 static struct config 
{ 
  60 static struct redisCommand cmdTable
[] = { 
  61     {"get",2,REDIS_CMD_INLINE
}, 
  62     {"set",3,REDIS_CMD_BULK
}, 
  63     {"setnx",3,REDIS_CMD_BULK
}, 
  64     {"del",-2,REDIS_CMD_INLINE
}, 
  65     {"exists",2,REDIS_CMD_INLINE
}, 
  66     {"incr",2,REDIS_CMD_INLINE
}, 
  67     {"decr",2,REDIS_CMD_INLINE
}, 
  68     {"rpush",3,REDIS_CMD_BULK
}, 
  69     {"lpush",3,REDIS_CMD_BULK
}, 
  70     {"rpop",2,REDIS_CMD_INLINE
}, 
  71     {"lpop",2,REDIS_CMD_INLINE
}, 
  72     {"llen",2,REDIS_CMD_INLINE
}, 
  73     {"lindex",3,REDIS_CMD_INLINE
}, 
  74     {"lset",4,REDIS_CMD_BULK
}, 
  75     {"lrange",4,REDIS_CMD_INLINE
}, 
  76     {"ltrim",4,REDIS_CMD_INLINE
}, 
  77     {"lrem",4,REDIS_CMD_BULK
}, 
  78     {"sadd",3,REDIS_CMD_BULK
}, 
  79     {"srem",3,REDIS_CMD_BULK
}, 
  80     {"smove",4,REDIS_CMD_BULK
}, 
  81     {"sismember",3,REDIS_CMD_BULK
}, 
  82     {"scard",2,REDIS_CMD_INLINE
}, 
  83     {"spop",2,REDIS_CMD_INLINE
}, 
  84     {"srandmember",2,REDIS_CMD_INLINE
}, 
  85     {"sinter",-2,REDIS_CMD_INLINE
}, 
  86     {"sinterstore",-3,REDIS_CMD_INLINE
}, 
  87     {"sunion",-2,REDIS_CMD_INLINE
}, 
  88     {"sunionstore",-3,REDIS_CMD_INLINE
}, 
  89     {"sdiff",-2,REDIS_CMD_INLINE
}, 
  90     {"sdiffstore",-3,REDIS_CMD_INLINE
}, 
  91     {"smembers",2,REDIS_CMD_INLINE
}, 
  92     {"zadd",4,REDIS_CMD_BULK
}, 
  93     {"zrem",3,REDIS_CMD_BULK
}, 
  94     {"zremrangebyscore",4,REDIS_CMD_INLINE
}, 
  95     {"zrange",4,REDIS_CMD_INLINE
}, 
  96     {"zrangebyscore",4,REDIS_CMD_INLINE
}, 
  97     {"zrevrange",4,REDIS_CMD_INLINE
}, 
  98     {"zcard",2,REDIS_CMD_INLINE
}, 
  99     {"zscore",3,REDIS_CMD_BULK
}, 
 100     {"incrby",3,REDIS_CMD_INLINE
}, 
 101     {"decrby",3,REDIS_CMD_INLINE
}, 
 102     {"getset",3,REDIS_CMD_BULK
}, 
 103     {"randomkey",1,REDIS_CMD_INLINE
}, 
 104     {"select",2,REDIS_CMD_INLINE
}, 
 105     {"move",3,REDIS_CMD_INLINE
}, 
 106     {"rename",3,REDIS_CMD_INLINE
}, 
 107     {"renamenx",3,REDIS_CMD_INLINE
}, 
 108     {"keys",2,REDIS_CMD_INLINE
}, 
 109     {"dbsize",1,REDIS_CMD_INLINE
}, 
 110     {"ping",1,REDIS_CMD_INLINE
}, 
 111     {"echo",2,REDIS_CMD_BULK
}, 
 112     {"save",1,REDIS_CMD_INLINE
}, 
 113     {"bgsave",1,REDIS_CMD_INLINE
}, 
 114     {"shutdown",1,REDIS_CMD_INLINE
}, 
 115     {"lastsave",1,REDIS_CMD_INLINE
}, 
 116     {"type",2,REDIS_CMD_INLINE
}, 
 117     {"flushdb",1,REDIS_CMD_INLINE
}, 
 118     {"flushall",1,REDIS_CMD_INLINE
}, 
 119     {"sort",-2,REDIS_CMD_INLINE
}, 
 120     {"info",1,REDIS_CMD_INLINE
}, 
 121     {"mget",-2,REDIS_CMD_INLINE
}, 
 122     {"expire",3,REDIS_CMD_INLINE
}, 
 123     {"ttl",2,REDIS_CMD_INLINE
}, 
 124     {"slaveof",3,REDIS_CMD_INLINE
}, 
 125     {"debug",-2,REDIS_CMD_INLINE
}, 
 126     {"mset",-3,REDIS_CMD_MULTIBULK
}, 
 127     {"msetnx",-3,REDIS_CMD_MULTIBULK
}, 
 131 static int cliReadReply(int fd
); 
 133 static struct redisCommand 
*lookupCommand(char *name
) { 
 135     while(cmdTable
[j
].name 
!= NULL
) { 
 136         if (!strcasecmp(name
,cmdTable
[j
].name
)) return &cmdTable
[j
]; 
 142 static int cliConnect(void) { 
 143     char err
[ANET_ERR_LEN
]; 
 146     fd 
= anetTcpConnect(err
,config
.hostip
,config
.hostport
); 
 147     if (fd 
== ANET_ERR
) { 
 148         fprintf(stderr
,"Connect: %s\n",err
); 
 151     anetTcpNoDelay(NULL
,fd
); 
 155 static sds 
cliReadLine(int fd
) { 
 156     sds line 
= sdsempty(); 
 166         } else if ((ret 
== 0) || (c 
== '\n')) { 
 169             line 
= sdscatlen(line
,&c
,1); 
 172     return sdstrim(line
,"\r\n"); 
 175 static int cliReadSingleLineReply(int fd
) { 
 176     sds reply 
= cliReadLine(fd
); 
 178     if (reply 
== NULL
) return 1; 
 179     printf("%s\n", reply
); 
 183 static int cliReadBulkReply(int fd
) { 
 184     sds replylen 
= cliReadLine(fd
); 
 185     char *reply
, crlf
[2]; 
 188     if (replylen 
== NULL
) return 1; 
 189     bulklen 
= atoi(replylen
); 
 195     reply 
= zmalloc(bulklen
); 
 196     anetRead(fd
,reply
,bulklen
); 
 198     if (bulklen 
&& fwrite(reply
,bulklen
,1,stdout
) == 0) { 
 202     if (isatty(fileno(stdout
)) && reply
[bulklen
-1] != '\n') 
 208 static int cliReadMultiBulkReply(int fd
) { 
 209     sds replylen 
= cliReadLine(fd
); 
 212     if (replylen 
== NULL
) return 1; 
 213     elements 
= atoi(replylen
); 
 214     if (elements 
== -1) { 
 220         printf("(empty list or set)\n"); 
 224         if (cliReadReply(fd
)) return 1; 
 230 static int cliReadReply(int fd
) { 
 233     if (anetRead(fd
,&type
,1) <= 0) exit(1); 
 237         cliReadSingleLineReply(fd
); 
 241         return cliReadSingleLineReply(fd
); 
 243         return cliReadBulkReply(fd
); 
 245         return cliReadMultiBulkReply(fd
); 
 247         printf("protocol error, got '%c' as reply type byte\n", type
); 
 252 static int cliSendCommand(int argc
, char **argv
) { 
 253     struct redisCommand 
*rc 
= lookupCommand(argv
[0]); 
 254     int fd
, j
, retval 
= 0; 
 255     sds cmd 
= sdsempty(); 
 258         fprintf(stderr
,"Unknown command '%s'\n",argv
[0]); 
 262     if ((rc
->arity 
> 0 && argc 
!= rc
->arity
) || 
 263         (rc
->arity 
< 0 && argc 
< -rc
->arity
)) { 
 264             fprintf(stderr
,"Wrong number of arguments for '%s'\n",rc
->name
); 
 267     if ((fd 
= cliConnect()) == -1) return 1; 
 269     /* Build the command to send */ 
 270     if (rc
->flags 
& REDIS_CMD_MULTIBULK
) { 
 271         cmd 
= sdscatprintf(cmd
,"*%d\r\n",argc
); 
 272         for (j 
= 0; j 
< argc
; j
++) { 
 273             cmd 
= sdscatprintf(cmd
,"$%d\r\n",sdslen(argv
[j
])); 
 274             cmd 
= sdscatlen(cmd
,argv
[j
],sdslen(argv
[j
])); 
 275             cmd 
= sdscatlen(cmd
,"\r\n",2); 
 278         for (j 
= 0; j 
< argc
; j
++) { 
 279             if (j 
!= 0) cmd 
= sdscat(cmd
," "); 
 280             if (j 
== argc
-1 && rc
->flags 
& REDIS_CMD_BULK
) { 
 281                 cmd 
= sdscatprintf(cmd
,"%d",sdslen(argv
[j
])); 
 283                 cmd 
= sdscatlen(cmd
,argv
[j
],sdslen(argv
[j
])); 
 286         cmd 
= sdscat(cmd
,"\r\n"); 
 287         if (rc
->flags 
& REDIS_CMD_BULK
) { 
 288             cmd 
= sdscatlen(cmd
,argv
[argc
-1],sdslen(argv
[argc
-1])); 
 289             cmd 
= sdscatlen(cmd
,"\r\n",2); 
 292     anetWrite(fd
,cmd
,sdslen(cmd
)); 
 293     retval 
= cliReadReply(fd
); 
 302 static int parseOptions(int argc
, char **argv
) { 
 305     for (i 
= 1; i 
< argc
; i
++) { 
 306         int lastarg 
= i
==argc
-1; 
 308         if (!strcmp(argv
[i
],"-h") && !lastarg
) { 
 309             char *ip 
= zmalloc(32); 
 310             if (anetResolve(NULL
,argv
[i
+1],ip
) == ANET_ERR
) { 
 311                 printf("Can't resolve %s\n", argv
[i
]); 
 316         } else if (!strcmp(argv
[i
],"-p") && !lastarg
) { 
 317             config
.hostport 
= atoi(argv
[i
+1]); 
 326 static sds 
readArgFromStdin(void) { 
 328     sds arg 
= sdsempty(); 
 331         int nread 
= read(fileno(stdin
),buf
,1024); 
 333         if (nread 
== 0) break; 
 334         else if (nread 
== -1) { 
 335             perror("Reading from standard input"); 
 338         arg 
= sdscatlen(arg
,buf
,nread
); 
 343 int main(int argc
, char **argv
) { 
 346     struct redisCommand 
*rc
; 
 348     config
.hostip 
= "127.0.0.1"; 
 349     config
.hostport 
= 6379; 
 351     firstarg 
= parseOptions(argc
,argv
); 
 355     /* Turn the plain C strings into Sds strings */ 
 356     argvcopy 
= zmalloc(sizeof(char*)*argc
+1); 
 357     for(j 
= 0; j 
< argc
; j
++) 
 358         argvcopy
[j
] = sdsnew(argv
[j
]); 
 361         fprintf(stderr
, "usage: redis-cli [-h host] [-p port] cmd arg1 arg2 arg3 ... argN\n"); 
 362         fprintf(stderr
, "usage: echo \"argN\" | redis-cli [-h host] [-p port] cmd arg1 arg2 ... arg(N-1)\n"); 
 363         fprintf(stderr
, "\nIf a pipe from standard input is detected this data is used as last argument.\n\n"); 
 364         fprintf(stderr
, "example: cat /etc/passwd | redis-cli set my_passwd\n"); 
 365         fprintf(stderr
, "example: redis-cli get my_passwd\n"); 
 369     /* Read the last argument from stdandard input if needed */ 
 370     if ((rc 
= lookupCommand(argv
[0])) != NULL
) { 
 371         if (rc
->arity 
> 0 && argc 
== rc
->arity
-1) { 
 372             sds lastarg 
= readArgFromStdin(); 
 373             argvcopy
[argc
] = lastarg
; 
 378     return cliSendCommand(argc
, argvcopy
);