First implementation of the ASKING command. Semantics still to verify.

This commit is contained in:
antirez 2011-10-17 17:35:23 +02:00
parent e0aab1fc79
commit 6856c7b4d6
4 changed files with 23 additions and 2 deletions

View File

@ -1657,6 +1657,19 @@ void dumpCommand(redisClient *c) {
return; return;
} }
/* The ASKING command is required after a -ASK redirection.
* The client should issue ASKING before to actualy send the command to
* the target instance. See the Redis Cluster specification for more
* information. */
void askingCommand(redisClient *c) {
if (server.cluster_enabled == 0) {
addReplyError(c,"This instance has cluster support disabled");
return;
}
c->flags |= REDIS_ASKING;
addReply(c,shared.ok);
}
/* ----------------------------------------------------------------------------- /* -----------------------------------------------------------------------------
* Cluster functions related to serving / redirecting clients * Cluster functions related to serving / redirecting clients
* -------------------------------------------------------------------------- */ * -------------------------------------------------------------------------- */
@ -1750,9 +1763,12 @@ clusterNode *getNodeByQuery(redisClient *c, struct redisCommand *cmd, robj **arg
} }
/* Handle the case in which we are receiving this hash slot from /* Handle the case in which we are receiving this hash slot from
* another instance, so we'll accept the query even if in the table * another instance, so we'll accept the query even if in the table
* it is assigned to a different node. */ * it is assigned to a different node, but only if the client
if (server.cluster.importing_slots_from[slot] != NULL) * issued an ASKING command before. */
if (server.cluster.importing_slots_from[slot] != NULL &&
c->flags & REDIS_ASKING) {
return server.cluster.myself; return server.cluster.myself;
}
/* It's not a -ASK case. Base case: just return the right node. */ /* It's not a -ASK case. Base case: just return the right node. */
return n; return n;
} }

View File

@ -625,6 +625,8 @@ void resetClient(redisClient *c) {
c->reqtype = 0; c->reqtype = 0;
c->multibulklen = 0; c->multibulklen = 0;
c->bulklen = -1; c->bulklen = -1;
/* We clear the ASKING flag as well if we are not inside a MULTI. */
if (!(c->flags & REDIS_MULTI)) c->flags &= (~REDIS_ASKING);
} }
void closeTimedoutClients(void) { void closeTimedoutClients(void) {

View File

@ -209,6 +209,7 @@ struct redisCommand redisCommandTable[] = {
{"cluster",clusterCommand,-2,"ar",0,NULL,0,0,0,0,0}, {"cluster",clusterCommand,-2,"ar",0,NULL,0,0,0,0,0},
{"restore",restoreCommand,4,"awm",0,NULL,1,1,1,0,0}, {"restore",restoreCommand,4,"awm",0,NULL,1,1,1,0,0},
{"migrate",migrateCommand,6,"aw",0,NULL,0,0,0,0,0}, {"migrate",migrateCommand,6,"aw",0,NULL,0,0,0,0,0},
{"asking",askingCommand,1,"r",0,NULL,0,0,0,0,0},
{"dump",dumpCommand,2,"ar",0,NULL,0,0,0,0,0}, {"dump",dumpCommand,2,"ar",0,NULL,0,0,0,0,0},
{"object",objectCommand,-2,"r",0,NULL,0,0,0,0,0}, {"object",objectCommand,-2,"r",0,NULL,0,0,0,0,0},
{"client",clientCommand,-2,"ar",0,NULL,0,0,0,0,0}, {"client",clientCommand,-2,"ar",0,NULL,0,0,0,0,0},

View File

@ -133,6 +133,7 @@
#define REDIS_UNBLOCKED 256 /* This client was unblocked and is stored in #define REDIS_UNBLOCKED 256 /* This client was unblocked and is stored in
server.unblocked_clients */ server.unblocked_clients */
#define REDIS_LUA_CLIENT 512 /* This is a non connected client used by Lua */ #define REDIS_LUA_CLIENT 512 /* This is a non connected client used by Lua */
#define REDIS_ASKING 1024 /* Client issued the ASKING command */
/* Client request types */ /* Client request types */
#define REDIS_REQ_INLINE 1 #define REDIS_REQ_INLINE 1
@ -1121,6 +1122,7 @@ void unwatchCommand(redisClient *c);
void clusterCommand(redisClient *c); void clusterCommand(redisClient *c);
void restoreCommand(redisClient *c); void restoreCommand(redisClient *c);
void migrateCommand(redisClient *c); void migrateCommand(redisClient *c);
void askingCommand(redisClient *c);
void dumpCommand(redisClient *c); void dumpCommand(redisClient *c);
void objectCommand(redisClient *c); void objectCommand(redisClient *c);
void clientCommand(redisClient *c); void clientCommand(redisClient *c);