]> git.saurik.com Git - redis.git/blob - src/sentinel.c
8ec5d151b99f6e14388b2d3e918aee5dc2d3dd6b
[redis.git] / src / sentinel.c
1 /* Redis Sentinel implementation
2 * -----------------------------
3 *
4 * Copyright (c) 2009-2012, Salvatore Sanfilippo <antirez at gmail dot com>
5 * All rights reserved.
6 *
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions are met:
9 *
10 * * Redistributions of source code must retain the above copyright notice,
11 * this list of conditions and the following disclaimer.
12 * * Redistributions in binary form must reproduce the above copyright
13 * notice, this list of conditions and the following disclaimer in the
14 * documentation and/or other materials provided with the distribution.
15 * * Neither the name of Redis nor the names of its contributors may be used
16 * to endorse or promote products derived from this software without
17 * specific prior written permission.
18 *
19 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
20 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
21 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
22 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
23 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
24 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
25 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
26 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
27 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
28 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
29 * POSSIBILITY OF SUCH DAMAGE.
30 */
31
32 #include "redis.h"
33 #include "hiredis.h"
34 #include "async.h"
35
36 #include <ctype.h>
37 #include <arpa/inet.h>
38 #include <sys/socket.h>
39
40 #define REDIS_SENTINEL_PORT 26379
41
42 /* ======================== Sentinel global state =========================== */
43
44 typedef long long mstime_t; /* millisecond time type. */
45
46 /* Address object, used to describe an ip:port pair. */
47 typedef struct sentinelAddr {
48 char *ip;
49 int port;
50 } sentinelAddr;
51
52 /* A Sentinel Redis Instance object is monitoring. */
53 #define SRI_MASTER (1<<0)
54 #define SRI_SLAVE (1<<1)
55 #define SRI_SENTINEL (1<<2)
56 #define SRI_DISCONNECTED (1<<3)
57 #define SRI_S_DOWN (1<<4) /* Subjectively down (no quorum). */
58 #define SRI_O_DOWN (1<<5) /* Objectively down (quorum reached). */
59 #define SRI_MASTER_DOWN (1<<6) /* A Sentinel with this flag set thinks that
60 its master is down. */
61 /* SRI_CAN_FAILOVER when set in an SRI_MASTER instance means that we are
62 * allowed to perform the failover for this master.
63 * When set in a SRI_SENTINEL instance means that sentinel is allowed to
64 * perform the failover on its master. */
65 #define SRI_CAN_FAILOVER (1<<7)
66 #define SRI_FAILOVER_IN_PROGRESS (1<<8) /* Failover is in progress for
67 this master. */
68 #define SRI_I_AM_THE_LEADER (1<<9) /* We are the leader for this master. */
69 #define SRI_PROMOTED (1<<10) /* Slave selected for promotion. */
70 #define SRI_RECONF_SENT (1<<11) /* SLAVEOF <newmaster> sent. */
71 #define SRI_RECONF_INPROG (1<<12) /* Slave synchronization in progress. */
72 #define SRI_RECONF_DONE (1<<13) /* Slave synchronized with new master. */
73
74 #define SENTINEL_INFO_PERIOD 10000
75 #define SENTINEL_PING_PERIOD 1000
76 #define SENTINEL_ASK_PERIOD 1000
77 #define SENTINEL_PUBLISH_PERIOD 5000
78 #define SENTINEL_DOWN_AFTER_PERIOD 30000
79 #define SENTINEL_HELLO_CHANNEL "__sentinel__:hello"
80 #define SENTINEL_TILT_TRIGGER 2000
81 #define SENTINEL_TILT_PERIOD (SENTINEL_PING_PERIOD*30)
82 #define SENTINEL_DEFAULT_SLAVE_PRIORITY 100
83 #define SENTINEL_PROMOTION_RETRY_PERIOD 30000
84 #define SENTINEL_SLAVE_RECONF_RETRY_PERIOD 10000
85 #define SENTINEL_DEFAULT_PARALLEL_SYNCS 1
86 #define SENTINEL_MIN_LINK_RECONNECT_PERIOD 15000
87 #define SENTINEL_DEFAULT_FAILOVER_TIMEOUT (60*15*1000)
88 #define SENTINEL_MAX_PENDING_COMMANDS 100
89 #define SENTINEL_EXTENDED_SDOWN_MULTIPLIER 10
90
91 /* How many milliseconds is an information valid? This applies for instance
92 * to the reply to SENTINEL IS-MASTER-DOWN-BY-ADDR replies. */
93 #define SENTINEL_INFO_VALIDITY_TIME 5000
94 #define SENTINEL_FAILOVER_FIXED_DELAY 5000
95 #define SENTINEL_FAILOVER_MAX_RANDOM_DELAY 10000
96
97 /* Failover machine different states. */
98 #define SENTINEL_FAILOVER_STATE_NONE 0 /* No failover in progress. */
99 #define SENTINEL_FAILOVER_STATE_WAIT_START 1 /* Wait for failover_start_time*/
100 #define SENTINEL_FAILOVER_STATE_SELECT_SLAVE 2 /* Select slave to promote */
101 #define SENTINEL_FAILOVER_STATE_SEND_SLAVEOF_NOONE 3 /* Slave -> Master */
102 #define SENTINEL_FAILOVER_STATE_WAIT_PROMOTION 4 /* Wait slave to change role */
103 #define SENTINEL_FAILOVER_STATE_RECONF_SLAVES 5 /* SLAVEOF newmaster */
104 #define SENTINEL_FAILOVER_STATE_WAIT_NEXT_SLAVE 6 /* wait replication */
105 #define SENTINEL_FAILOVER_STATE_ALERT_CLIENTS 7 /* Run user script. */
106 #define SENTINEL_FAILOVER_STATE_WAIT_ALERT_SCRIPT 8 /* Wait script exec. */
107 #define SENTINEL_FAILOVER_STATE_DETECT_END 9 /* Check for failover end. */
108 #define SENTINEL_FAILOVER_STATE_UPDATE_CONFIG 10 /* Monitor promoted slave. */
109
110 #define SENTINEL_MASTER_LINK_STATUS_UP 0
111 #define SENTINEL_MASTER_LINK_STATUS_DOWN 1
112
113 /* Generic flags that can be used with different functions. */
114 #define SENTINEL_NO_FLAGS 0
115 #define SENTINEL_GENERATE_EVENT 1
116
117 typedef struct sentinelRedisInstance {
118 int flags; /* See SRI_... defines */
119 char *name; /* Master name from the point of view of this sentinel. */
120 char *runid; /* run ID of this instance. */
121 sentinelAddr *addr; /* Master host. */
122 redisAsyncContext *cc; /* Hiredis context for commands. */
123 redisAsyncContext *pc; /* Hiredis context for Pub / Sub. */
124 int pending_commands; /* Number of commands sent waiting for a reply. */
125 mstime_t cc_conn_time; /* cc connection time. */
126 mstime_t pc_conn_time; /* pc connection time. */
127 mstime_t pc_last_activity; /* Last time we received any message. */
128 mstime_t last_avail_time; /* Last time the instance replied to ping with
129 a reply we consider valid. */
130 mstime_t last_pong_time; /* Last time the instance replied to ping,
131 whatever the reply was. That's used to check
132 if the link is idle and must be reconnected. */
133 mstime_t last_pub_time; /* Last time we sent hello via Pub/Sub. */
134 mstime_t last_hello_time; /* Only used if SRI_SENTINEL is set. Last time
135 we received an hello from this Sentinel
136 via Pub/Sub. */
137 mstime_t last_master_down_reply_time; /* Time of last reply to
138 SENTINEL is-master-down command. */
139 mstime_t s_down_since_time; /* Subjectively down since time. */
140 mstime_t o_down_since_time; /* Objectively down since time. */
141 mstime_t down_after_period; /* Consider it down after that period. */
142 mstime_t info_refresh; /* Time at which we received INFO output from it. */
143
144 /* Master specific. */
145 dict *sentinels; /* Other sentinels monitoring the same master. */
146 dict *slaves; /* Slaves for this master instance. */
147 int quorum; /* Number of sentinels that need to agree on failure. */
148 int parallel_syncs; /* How many slaves to reconfigure at same time. */
149
150 /* Slave specific. */
151 mstime_t master_link_down_time; /* Slave replication link down time. */
152 int slave_priority; /* Slave priority according to its INFO output. */
153 mstime_t slave_reconf_sent_time; /* Time at which we sent SLAVE OF <new> */
154 struct sentinelRedisInstance *master; /* Master instance if SRI_SLAVE is set. */
155 char *slave_master_host; /* Master host as reported by INFO */
156 int slave_master_port; /* Master port as reported by INFO */
157 int slave_master_link_status; /* Master link status as reported by INFO */
158 /* Failover */
159 char *leader; /* If this is a master instance, this is the runid of
160 the Sentinel that should perform the failover. If
161 this is a Sentinel, this is the runid of the Sentinel
162 that this other Sentinel is voting as leader.
163 This field is valid only if SRI_MASTER_DOWN is
164 set on the Sentinel instance. */
165 int failover_state; /* See SENTINEL_FAILOVER_STATE_* defines. */
166 mstime_t failover_state_change_time;
167 mstime_t failover_start_time; /* When to start to failover if leader. */
168 mstime_t failover_timeout; /* Max time to refresh failover state. */
169 struct sentinelRedisInstance *promoted_slave; /* Promoted slave instance. */
170 /* Scripts executed to notify admin or reconfigure clients: when they
171 * are set to NULL no script is executed. */
172 char *notify_script;
173 char *client_reconfig_script;
174 } sentinelRedisInstance;
175
176 /* Main state. */
177 struct sentinelState {
178 dict *masters; /* Dictionary of master sentinelRedisInstances.
179 Key is the instance name, value is the
180 sentinelRedisInstance structure pointer. */
181 int tilt; /* Are we in TILT mode? */
182 mstime_t tilt_start_time; /* When TITL started. */
183 mstime_t previous_time; /* Time last time we ran the time handler. */
184 } sentinel;
185
186 /* ======================= hiredis ae.c adapters =============================
187 * Note: this implementation is taken from hiredis/adapters/ae.h, however
188 * we have our modified copy for Sentinel in order to use our allocator
189 * and to have full control over how the adapter works. */
190
191 typedef struct redisAeEvents {
192 redisAsyncContext *context;
193 aeEventLoop *loop;
194 int fd;
195 int reading, writing;
196 } redisAeEvents;
197
198 static void redisAeReadEvent(aeEventLoop *el, int fd, void *privdata, int mask) {
199 ((void)el); ((void)fd); ((void)mask);
200
201 redisAeEvents *e = (redisAeEvents*)privdata;
202 redisAsyncHandleRead(e->context);
203 }
204
205 static void redisAeWriteEvent(aeEventLoop *el, int fd, void *privdata, int mask) {
206 ((void)el); ((void)fd); ((void)mask);
207
208 redisAeEvents *e = (redisAeEvents*)privdata;
209 redisAsyncHandleWrite(e->context);
210 }
211
212 static void redisAeAddRead(void *privdata) {
213 redisAeEvents *e = (redisAeEvents*)privdata;
214 aeEventLoop *loop = e->loop;
215 if (!e->reading) {
216 e->reading = 1;
217 aeCreateFileEvent(loop,e->fd,AE_READABLE,redisAeReadEvent,e);
218 }
219 }
220
221 static void redisAeDelRead(void *privdata) {
222 redisAeEvents *e = (redisAeEvents*)privdata;
223 aeEventLoop *loop = e->loop;
224 if (e->reading) {
225 e->reading = 0;
226 aeDeleteFileEvent(loop,e->fd,AE_READABLE);
227 }
228 }
229
230 static void redisAeAddWrite(void *privdata) {
231 redisAeEvents *e = (redisAeEvents*)privdata;
232 aeEventLoop *loop = e->loop;
233 if (!e->writing) {
234 e->writing = 1;
235 aeCreateFileEvent(loop,e->fd,AE_WRITABLE,redisAeWriteEvent,e);
236 }
237 }
238
239 static void redisAeDelWrite(void *privdata) {
240 redisAeEvents *e = (redisAeEvents*)privdata;
241 aeEventLoop *loop = e->loop;
242 if (e->writing) {
243 e->writing = 0;
244 aeDeleteFileEvent(loop,e->fd,AE_WRITABLE);
245 }
246 }
247
248 static void redisAeCleanup(void *privdata) {
249 redisAeEvents *e = (redisAeEvents*)privdata;
250 redisAeDelRead(privdata);
251 redisAeDelWrite(privdata);
252 zfree(e);
253 }
254
255 static int redisAeAttach(aeEventLoop *loop, redisAsyncContext *ac) {
256 redisContext *c = &(ac->c);
257 redisAeEvents *e;
258
259 /* Nothing should be attached when something is already attached */
260 if (ac->ev.data != NULL)
261 return REDIS_ERR;
262
263 /* Create container for context and r/w events */
264 e = (redisAeEvents*)zmalloc(sizeof(*e));
265 e->context = ac;
266 e->loop = loop;
267 e->fd = c->fd;
268 e->reading = e->writing = 0;
269
270 /* Register functions to start/stop listening for events */
271 ac->ev.addRead = redisAeAddRead;
272 ac->ev.delRead = redisAeDelRead;
273 ac->ev.addWrite = redisAeAddWrite;
274 ac->ev.delWrite = redisAeDelWrite;
275 ac->ev.cleanup = redisAeCleanup;
276 ac->ev.data = e;
277
278 return REDIS_OK;
279 }
280
281 /* ============================= Prototypes ================================= */
282
283 void sentinelLinkEstablishedCallback(const redisAsyncContext *c, int status);
284 void sentinelDisconnectCallback(const redisAsyncContext *c, int status);
285 void sentinelReceiveHelloMessages(redisAsyncContext *c, void *reply, void *privdata);
286 sentinelRedisInstance *sentinelGetMasterByName(char *name);
287 char *sentinelGetSubjectiveLeader(sentinelRedisInstance *master);
288 char *sentinelGetObjectiveLeader(sentinelRedisInstance *master);
289 int yesnotoi(char *s);
290 void sentinelDisconnectInstanceFromContext(const redisAsyncContext *c);
291 void sentinelKillLink(sentinelRedisInstance *ri, redisAsyncContext *c);
292 const char *sentinelRedisInstanceTypeStr(sentinelRedisInstance *ri);
293
294 /* ========================= Dictionary types =============================== */
295
296 unsigned int dictSdsHash(const void *key);
297 int dictSdsKeyCompare(void *privdata, const void *key1, const void *key2);
298 void releaseSentinelRedisInstance(sentinelRedisInstance *ri);
299
300 void dictInstancesValDestructor (void *privdata, void *obj) {
301 releaseSentinelRedisInstance(obj);
302 }
303
304 /* Instance name (sds) -> instance (sentinelRedisInstance pointer)
305 *
306 * also used for: sentinelRedisInstance->sentinels dictionary that maps
307 * sentinels ip:port to last seen time in Pub/Sub hello message. */
308 dictType instancesDictType = {
309 dictSdsHash, /* hash function */
310 NULL, /* key dup */
311 NULL, /* val dup */
312 dictSdsKeyCompare, /* key compare */
313 NULL, /* key destructor */
314 dictInstancesValDestructor /* val destructor */
315 };
316
317 /* Instance runid (sds) -> votes (long casted to void*)
318 *
319 * This is useful into sentinelGetObjectiveLeader() function in order to
320 * count the votes and understand who is the leader. */
321 dictType leaderVotesDictType = {
322 dictSdsHash, /* hash function */
323 NULL, /* key dup */
324 NULL, /* val dup */
325 dictSdsKeyCompare, /* key compare */
326 NULL, /* key destructor */
327 NULL /* val destructor */
328 };
329
330 /* =========================== Initialization =============================== */
331
332 void sentinelCommand(redisClient *c);
333
334 struct redisCommand sentinelcmds[] = {
335 {"ping",pingCommand,1,"",0,NULL,0,0,0,0,0},
336 {"sentinel",sentinelCommand,-2,"",0,NULL,0,0,0,0,0},
337 {"subscribe",subscribeCommand,-2,"",0,NULL,0,0,0,0,0},
338 {"unsubscribe",unsubscribeCommand,-1,"",0,NULL,0,0,0,0,0},
339 {"psubscribe",psubscribeCommand,-2,"",0,NULL,0,0,0,0,0},
340 {"punsubscribe",punsubscribeCommand,-1,"",0,NULL,0,0,0,0,0}
341 };
342
343 /* This function overwrites a few normal Redis config default with Sentinel
344 * specific defaults. */
345 void initSentinelConfig(void) {
346 server.port = REDIS_SENTINEL_PORT;
347 }
348
349 /* Perform the Sentinel mode initialization. */
350 void initSentinel(void) {
351 int j;
352
353 /* Remove usual Redis commands from the command table, then just add
354 * the SENTINEL command. */
355 dictEmpty(server.commands);
356 for (j = 0; j < sizeof(sentinelcmds)/sizeof(sentinelcmds[0]); j++) {
357 int retval;
358 struct redisCommand *cmd = sentinelcmds+j;
359
360 retval = dictAdd(server.commands, sdsnew(cmd->name), cmd);
361 redisAssert(retval == DICT_OK);
362 }
363
364 /* Initialize various data structures. */
365 sentinel.masters = dictCreate(&instancesDictType,NULL);
366 sentinel.tilt = 0;
367 sentinel.tilt_start_time = mstime();
368 sentinel.previous_time = mstime();
369 }
370
371 /* ============================== sentinelAddr ============================== */
372
373 /* Create a sentinelAddr object and return it on success.
374 * On error NULL is returned and errno is set to:
375 * ENOENT: Can't resolve the hostname.
376 * EINVAL: Invalid port number.
377 */
378 sentinelAddr *createSentinelAddr(char *hostname, int port) {
379 char buf[32];
380 sentinelAddr *sa;
381
382 if (port <= 0 || port > 65535) {
383 errno = EINVAL;
384 return NULL;
385 }
386 if (anetResolve(NULL,hostname,buf) == ANET_ERR) {
387 errno = ENOENT;
388 return NULL;
389 }
390 sa = zmalloc(sizeof(*sa));
391 sa->ip = sdsnew(buf);
392 sa->port = port;
393 return sa;
394 }
395
396 /* Free a Sentinel address. Can't fail. */
397 void releaseSentinelAddr(sentinelAddr *sa) {
398 sdsfree(sa->ip);
399 zfree(sa);
400 }
401
402 /* =========================== Events notification ========================== */
403
404 void sentinelCallNotificationScript(char *scriptpath, char *type, char *msg) {
405 /* TODO: implement it. */
406 }
407
408 /* Send an event to log, pub/sub, user notification script.
409 *
410 * 'level' is the log level for logging. Only REDIS_WARNING events will trigger
411 * the execution of the user notification script.
412 *
413 * 'type' is the message type, also used as a pub/sub channel name.
414 *
415 * 'ri', is the redis instance target of this event if applicable, and is
416 * used to obtain the path of the notification script to execute.
417 *
418 * The remaining arguments are printf-alike.
419 * If the format specifier starts with the two characters "%@" then ri is
420 * not NULL, and the message is prefixed with an instance identifier in the
421 * following format:
422 *
423 * <instance type> <instance name> <ip> <port>
424 *
425 * If the instance type is not master, than the additional string is
426 * added to specify the originating master:
427 *
428 * @ <master name> <master ip> <master port>
429 *
430 * Any other specifier after "%@" is processed by printf itself.
431 */
432 void sentinelEvent(int level, char *type, sentinelRedisInstance *ri,
433 const char *fmt, ...) {
434 va_list ap;
435 char msg[REDIS_MAX_LOGMSG_LEN];
436 robj *channel, *payload;
437
438 /* Handle %@ */
439 if (fmt[0] == '%' && fmt[1] == '@') {
440 sentinelRedisInstance *master = (ri->flags & SRI_MASTER) ?
441 NULL : ri->master;
442
443 if (master) {
444 snprintf(msg, sizeof(msg), "%s %s %s %d @ %s %s %d",
445 sentinelRedisInstanceTypeStr(ri),
446 ri->name, ri->addr->ip, ri->addr->port,
447 master->name, master->addr->ip, master->addr->port);
448 } else {
449 snprintf(msg, sizeof(msg), "%s %s %s %d",
450 sentinelRedisInstanceTypeStr(ri),
451 ri->name, ri->addr->ip, ri->addr->port);
452 }
453 fmt += 2;
454 } else {
455 msg[0] = '\0';
456 }
457
458 /* Use vsprintf for the rest of the formatting if any. */
459 if (fmt[0] != '\0') {
460 va_start(ap, fmt);
461 vsnprintf(msg+strlen(msg), sizeof(msg)-strlen(msg), fmt, ap);
462 va_end(ap);
463 }
464
465 /* Log the message if the log level allows it to be logged. */
466 if (level >= server.verbosity)
467 redisLog(level,"%s %s",type,msg);
468
469 /* Publish the message via Pub/Sub if it's not a debugging one. */
470 if (level != REDIS_DEBUG) {
471 channel = createStringObject(type,strlen(type));
472 payload = createStringObject(msg,strlen(msg));
473 pubsubPublishMessage(channel,payload);
474 decrRefCount(channel);
475 decrRefCount(payload);
476 }
477
478 /* Call the notification script if applicable. */
479 if (level == REDIS_WARNING && ri != NULL) {
480 sentinelRedisInstance *master = (ri->flags & SRI_MASTER) ?
481 ri : ri->master;
482 if (master->notify_script) {
483 sentinelCallNotificationScript(master->notify_script,type,msg);
484 }
485 }
486 }
487
488 /* ========================== sentinelRedisInstance ========================= */
489
490 /* Create a redis instance, the following fields must be populated by the
491 * caller if needed:
492 * runid: set to NULL but will be populated once INFO output is received.
493 * info_refresh: is set to 0 to mean that we never received INFO so far.
494 *
495 * If SRI_MASTER is set into initial flags the instance is added to
496 * sentinel.masters table.
497 *
498 * if SRI_SLAVE or SRI_SENTINEL is set then 'master' must be not NULL and the
499 * instance is added into master->slaves or master->sentinels table.
500 *
501 * If the instance is a slave or sentinel, the name parameter is ignored and
502 * is created automatically as hostname:port.
503 *
504 * The function fails if hostname can't be resolved or port is out of range.
505 * When this happens NULL is returned and errno is set accordingly to the
506 * createSentinelAddr() function.
507 *
508 * The function may also fail and return NULL with errno set to EBUSY if
509 * a master or slave with the same name already exists. */
510 sentinelRedisInstance *createSentinelRedisInstance(char *name, int flags, char *hostname, int port, int quorum, sentinelRedisInstance *master) {
511 sentinelRedisInstance *ri;
512 sentinelAddr *addr;
513 dict *table;
514 char slavename[128], *sdsname;
515
516 redisAssert(flags & (SRI_MASTER|SRI_SLAVE|SRI_SENTINEL));
517 redisAssert((flags & SRI_MASTER) || master != NULL);
518
519 /* Check address validity. */
520 addr = createSentinelAddr(hostname,port);
521 if (addr == NULL) return NULL;
522
523 /* For slaves and sentinel we use ip:port as name. */
524 if (flags & (SRI_SLAVE|SRI_SENTINEL)) {
525 snprintf(slavename,sizeof(slavename),"%s:%d",hostname,port);
526 name = slavename;
527 }
528
529 /* Make sure the entry is not duplicated. This may happen when the same
530 * name for a master is used multiple times inside the configuration or
531 * if we try to add multiple times a slave or sentinel with same ip/port
532 * to a master. */
533 if (flags & SRI_MASTER) table = sentinel.masters;
534 else if (flags & SRI_SLAVE) table = master->slaves;
535 else if (flags & SRI_SENTINEL) table = master->sentinels;
536 sdsname = sdsnew(name);
537 if (dictFind(table,sdsname)) {
538 sdsfree(sdsname);
539 errno = EBUSY;
540 return NULL;
541 }
542
543 /* Create the instance object. */
544 ri = zmalloc(sizeof(*ri));
545 /* Note that all the instances are started in the disconnected state,
546 * the event loop will take care of connecting them. */
547 ri->flags = flags | SRI_DISCONNECTED;
548 ri->name = sdsname;
549 ri->runid = NULL;
550 ri->addr = addr;
551 ri->cc = NULL;
552 ri->pc = NULL;
553 ri->pending_commands = 0;
554 ri->cc_conn_time = 0;
555 ri->pc_conn_time = 0;
556 ri->pc_last_activity = 0;
557 ri->last_avail_time = mstime();
558 ri->last_pong_time = mstime();
559 ri->last_pub_time = mstime();
560 ri->last_hello_time = mstime();
561 ri->last_master_down_reply_time = mstime();
562 ri->s_down_since_time = 0;
563 ri->o_down_since_time = 0;
564 ri->down_after_period = master ? master->down_after_period :
565 SENTINEL_DOWN_AFTER_PERIOD;
566 ri->master_link_down_time = 0;
567 ri->slave_priority = SENTINEL_DEFAULT_SLAVE_PRIORITY;
568 ri->slave_reconf_sent_time = 0;
569 ri->slave_master_host = NULL;
570 ri->slave_master_port = 0;
571 ri->slave_master_link_status = SENTINEL_MASTER_LINK_STATUS_DOWN;
572 ri->sentinels = dictCreate(&instancesDictType,NULL);
573 ri->quorum = quorum;
574 ri->parallel_syncs = SENTINEL_DEFAULT_PARALLEL_SYNCS;
575 ri->master = master;
576 ri->slaves = dictCreate(&instancesDictType,NULL);
577 ri->info_refresh = 0;
578
579 /* Failover state. */
580 ri->leader = NULL;
581 ri->failover_state = SENTINEL_FAILOVER_STATE_NONE;
582 ri->failover_state_change_time = 0;
583 ri->failover_start_time = 0;
584 ri->failover_timeout = SENTINEL_DEFAULT_FAILOVER_TIMEOUT;
585 ri->promoted_slave = NULL;
586 ri->notify_script = NULL;
587 ri->client_reconfig_script = NULL;
588
589 /* Add into the right table. */
590 dictAdd(table, ri->name, ri);
591 return ri;
592 }
593
594 /* Release this instance and all its slaves, sentinels, hiredis connections.
595 * This function also takes care of unlinking the instance from the main
596 * masters table (if it is a master) or from its master sentinels/slaves table
597 * if it is a slave or sentinel. */
598 void releaseSentinelRedisInstance(sentinelRedisInstance *ri) {
599 /* Release all its slaves or sentinels if any. */
600 dictRelease(ri->sentinels);
601 dictRelease(ri->slaves);
602
603 /* Release hiredis connections. */
604 if (ri->cc) sentinelKillLink(ri,ri->cc);
605 if (ri->pc) sentinelKillLink(ri,ri->pc);
606
607 /* Free other resources. */
608 sdsfree(ri->name);
609 sdsfree(ri->runid);
610 sdsfree(ri->notify_script);
611 sdsfree(ri->client_reconfig_script);
612 sdsfree(ri->slave_master_host);
613 sdsfree(ri->leader);
614 releaseSentinelAddr(ri->addr);
615
616 /* Clear state into the master if needed. */
617 if ((ri->flags & SRI_SLAVE) && (ri->flags & SRI_PROMOTED) && ri->master)
618 ri->master->promoted_slave = NULL;
619
620 zfree(ri);
621 }
622
623 /* Lookup a slave in a master Redis instance, by ip and port. */
624 sentinelRedisInstance *sentinelRedisInstanceLookupSlave(
625 sentinelRedisInstance *ri, char *ip, int port)
626 {
627 sds key;
628 sentinelRedisInstance *slave;
629
630 redisAssert(ri->flags & SRI_MASTER);
631 key = sdscatprintf(sdsempty(),"%s:%d",ip,port);
632 slave = dictFetchValue(ri->slaves,key);
633 sdsfree(key);
634 return slave;
635 }
636
637 /* Return the name of the type of the instance as a string. */
638 const char *sentinelRedisInstanceTypeStr(sentinelRedisInstance *ri) {
639 if (ri->flags & SRI_MASTER) return "master";
640 else if (ri->flags & SRI_SLAVE) return "slave";
641 else if (ri->flags & SRI_SENTINEL) return "sentinel";
642 else return "unknown";
643 }
644
645 /* This function removes all the instances found in the dictionary of instances
646 * 'd', having either:
647 *
648 * 1) The same ip/port as specified.
649 * 2) The same runid.
650 *
651 * "1" and "2" don't need to verify at the same time, just one is enough.
652 * If "runid" is NULL it is not checked.
653 * Similarly if "ip" is NULL it is not checked.
654 *
655 * This function is useful because every time we add a new Sentinel into
656 * a master's Sentinels dictionary, we want to be very sure about not
657 * having duplicated instances for any reason. This is so important because
658 * we use those other sentinels in order to run our quorum protocol to
659 * understand if it's time to proceeed with the fail over.
660 *
661 * Making sure no duplication is possible we greately improve the robustness
662 * of the quorum (otherwise we may end counting the same instance multiple
663 * times for some reason).
664 *
665 * The function returns the number of Sentinels removed. */
666 int removeMatchingSentinelsFromMaster(sentinelRedisInstance *master, char *ip, int port, char *runid) {
667 dictIterator *di;
668 dictEntry *de;
669 int removed = 0;
670
671 di = dictGetSafeIterator(master->sentinels);
672 while((de = dictNext(di)) != NULL) {
673 sentinelRedisInstance *ri = dictGetVal(de);
674
675 if ((ri->runid && runid && strcmp(ri->runid,runid) == 0) ||
676 (ip && strcmp(ri->addr->ip,ip) == 0 && port == ri->addr->port))
677 {
678 dictDelete(master->sentinels,ri->name);
679 removed++;
680 }
681 }
682 dictReleaseIterator(di);
683 return removed;
684 }
685
686 /* Search an instance with the same runid, ip and port into a dictionary
687 * of instances. Return NULL if not found, otherwise return the instance
688 * pointer.
689 *
690 * runid or ip can be NULL. In such a case the search is performed only
691 * by the non-NULL field. */
692 sentinelRedisInstance *getSentinelRedisInstanceByAddrAndRunID(dict *instances, char *ip, int port, char *runid) {
693 dictIterator *di;
694 dictEntry *de;
695 sentinelRedisInstance *instance = NULL;
696
697 redisAssert(ip || runid); /* User must pass at least one search param. */
698 di = dictGetIterator(instances);
699 while((de = dictNext(di)) != NULL) {
700 sentinelRedisInstance *ri = dictGetVal(de);
701
702 if (runid && !ri->runid) continue;
703 if ((runid == NULL || strcmp(ri->runid, runid) == 0) &&
704 (ip == NULL || (strcmp(ri->addr->ip, ip) == 0 &&
705 ri->addr->port == port)))
706 {
707 instance = ri;
708 break;
709 }
710 }
711 dictReleaseIterator(di);
712 return instance;
713 }
714
715 /* Simple master lookup by name */
716 sentinelRedisInstance *sentinelGetMasterByName(char *name) {
717 sentinelRedisInstance *ri;
718 sds sdsname = sdsnew(name);
719
720 ri = dictFetchValue(sentinel.masters,sdsname);
721 sdsfree(sdsname);
722 return ri;
723 }
724
725 /* Add the specified flags to all the instances in the specified dictionary. */
726 void sentinelAddFlagsToDictOfRedisInstances(dict *instances, int flags) {
727 dictIterator *di;
728 dictEntry *de;
729
730 di = dictGetIterator(instances);
731 while((de = dictNext(di)) != NULL) {
732 sentinelRedisInstance *ri = dictGetVal(de);
733 ri->flags |= flags;
734 }
735 dictReleaseIterator(di);
736 }
737
738 /* Remove the specified flags to all the instances in the specified
739 * dictionary. */
740 void sentinelDelFlagsToDictOfRedisInstances(dict *instances, int flags) {
741 dictIterator *di;
742 dictEntry *de;
743
744 di = dictGetIterator(instances);
745 while((de = dictNext(di)) != NULL) {
746 sentinelRedisInstance *ri = dictGetVal(de);
747 ri->flags &= ~flags;
748 }
749 dictReleaseIterator(di);
750 }
751
752 /* Reset the state of a monitored master:
753 * 1) Remove all slaves.
754 * 2) Remove all sentinels.
755 * 3) Remove most of the flags resulting from runtime operations.
756 * 4) Reset timers to their default value.
757 * 5) In the process of doing this undo the failover if in progress.
758 * 6) Disconnect the connections with the master (will reconnect automatically).
759 */
760 void sentinelResetMaster(sentinelRedisInstance *ri, int flags) {
761 redisAssert(ri->flags & SRI_MASTER);
762 dictRelease(ri->slaves);
763 dictRelease(ri->sentinels);
764 ri->slaves = dictCreate(&instancesDictType,NULL);
765 ri->sentinels = dictCreate(&instancesDictType,NULL);
766 if (ri->cc) sentinelKillLink(ri,ri->cc);
767 if (ri->pc) sentinelKillLink(ri,ri->pc);
768 ri->flags &= SRI_MASTER|SRI_CAN_FAILOVER|SRI_DISCONNECTED;
769 if (ri->leader) {
770 sdsfree(ri->leader);
771 ri->leader = NULL;
772 }
773 ri->failover_state = SENTINEL_FAILOVER_STATE_NONE;
774 ri->failover_state_change_time = 0;
775 ri->failover_start_time = 0;
776 ri->promoted_slave = NULL;
777 sdsfree(ri->runid);
778 sdsfree(ri->slave_master_host);
779 ri->runid = NULL;
780 ri->slave_master_host = NULL;
781 ri->last_avail_time = mstime();
782 ri->last_pong_time = mstime();
783 if (flags & SENTINEL_GENERATE_EVENT)
784 sentinelEvent(REDIS_WARNING,"+reset-master",ri,"%@");
785 }
786
787 /* Call sentinelResetMaster() on every master with a name matching the specified
788 * pattern. */
789 int sentinelResetMastersByPattern(char *pattern, int flags) {
790 dictIterator *di;
791 dictEntry *de;
792 int reset = 0;
793
794 di = dictGetIterator(sentinel.masters);
795 while((de = dictNext(di)) != NULL) {
796 sentinelRedisInstance *ri = dictGetVal(de);
797
798 if (ri->name) {
799 if (stringmatch(pattern,ri->name,0)) {
800 sentinelResetMaster(ri,flags);
801 reset++;
802 }
803 }
804 }
805 dictReleaseIterator(di);
806 return reset;
807 }
808
809 /* Reset the specified master with sentinelResetMaster(), and also change
810 * the ip:port address, but take the name of the instance unmodified.
811 *
812 * This is used to handle the +switch-master and +redirect-to-master events.
813 *
814 * The function returns REDIS_ERR if the address can't be resolved for some
815 * reason. Otherwise REDIS_OK is returned.
816 *
817 * TODO: make this reset so that original sentinels are re-added with
818 * same ip / port / runid.
819 */
820
821 int sentinelResetMasterAndChangeAddress(sentinelRedisInstance *master, char *ip, int port) {
822 sentinelAddr *oldaddr, *newaddr;
823
824 newaddr = createSentinelAddr(ip,port);
825 if (newaddr == NULL) return REDIS_ERR;
826 sentinelResetMaster(master,SENTINEL_NO_FLAGS);
827 oldaddr = master->addr;
828 master->addr = newaddr;
829 /* Release the old address at the end so we are safe even if the function
830 * gets the master->addr->ip and master->addr->port as arguments. */
831 releaseSentinelAddr(oldaddr);
832 return REDIS_OK;
833 }
834
835 /* ============================ Config handling ============================= */
836 char *sentinelHandleConfiguration(char **argv, int argc) {
837 sentinelRedisInstance *ri;
838
839 if (!strcasecmp(argv[0],"monitor") && argc == 5) {
840 /* monitor <name> <host> <port> <quorum> */
841 int quorum = atoi(argv[4]);
842
843 if (quorum <= 0) return "Quorum must be 1 or greater.";
844 if (createSentinelRedisInstance(argv[1],SRI_MASTER,argv[2],
845 atoi(argv[3]),quorum,NULL) == NULL)
846 {
847 switch(errno) {
848 case EBUSY: return "Duplicated master name.";
849 case ENOENT: return "Can't resolve master instance hostname.";
850 case EINVAL: return "Invalid port number";
851 }
852 }
853 } else if (!strcasecmp(argv[0],"down-after-milliseconds") && argc == 3) {
854 /* down-after-milliseconds <name> <milliseconds> */
855 ri = sentinelGetMasterByName(argv[1]);
856 if (!ri) return "No such master with specified name.";
857 ri->down_after_period = atoi(argv[2]);
858 if (ri->down_after_period <= 0)
859 return "negative or zero time parameter.";
860 } else if (!strcasecmp(argv[0],"failover-timeout") && argc == 3) {
861 /* failover-timeout <name> <milliseconds> */
862 ri = sentinelGetMasterByName(argv[1]);
863 if (!ri) return "No such master with specified name.";
864 ri->failover_timeout = atoi(argv[2]);
865 if (ri->failover_timeout <= 0)
866 return "negative or zero time parameter.";
867 } else if (!strcasecmp(argv[0],"can-failover") && argc == 3) {
868 /* can-failover <name> <yes/no> */
869 int yesno = yesnotoi(argv[2]);
870
871 ri = sentinelGetMasterByName(argv[1]);
872 if (!ri) return "No such master with specified name.";
873 if (yesno == -1) return "Argument must be either yes or no.";
874 if (yesno)
875 ri->flags |= SRI_CAN_FAILOVER;
876 else
877 ri->flags &= ~SRI_CAN_FAILOVER;
878 } else if (!strcasecmp(argv[0],"parallel-syncs") && argc == 3) {
879 /* parallel-syncs <name> <milliseconds> */
880 ri = sentinelGetMasterByName(argv[1]);
881 if (!ri) return "No such master with specified name.";
882 ri->parallel_syncs = atoi(argv[2]);
883 } else {
884 return "Unrecognized sentinel configuration statement.";
885 }
886 return NULL;
887 }
888
889 /* ====================== hiredis connection handling ======================= */
890
891 /* Completely disconnect an hiredis link from an instance. */
892 void sentinelKillLink(sentinelRedisInstance *ri, redisAsyncContext *c) {
893 if (ri->cc == c) {
894 ri->cc = NULL;
895 ri->pending_commands = 0;
896 }
897 if (ri->pc == c) ri->pc = NULL;
898 c->data = NULL;
899 ri->flags |= SRI_DISCONNECTED;
900 redisAsyncFree(c);
901 }
902
903 /* This function takes an hiredis context that is in an error condition
904 * and make sure to mark the instance as disconnected performing the
905 * cleanup needed.
906 *
907 * Note: we don't free the hiredis context as hiredis will do it for us
908 * for async conenctions. */
909 void sentinelDisconnectInstanceFromContext(const redisAsyncContext *c) {
910 sentinelRedisInstance *ri = c->data;
911 int pubsub;
912
913 if (ri == NULL) return; /* The instance no longer exists. */
914
915 pubsub = (ri->pc == c);
916 sentinelEvent(REDIS_DEBUG, pubsub ? "-pubsub-link" : "-cmd-link", ri,
917 "%@ #%s", c->errstr);
918 if (pubsub)
919 ri->pc = NULL;
920 else
921 ri->cc = NULL;
922 ri->flags |= SRI_DISCONNECTED;
923 }
924
925 void sentinelLinkEstablishedCallback(const redisAsyncContext *c, int status) {
926 if (status != REDIS_OK) {
927 sentinelDisconnectInstanceFromContext(c);
928 } else {
929 sentinelRedisInstance *ri = c->data;
930 int pubsub = (ri->pc == c);
931
932 sentinelEvent(REDIS_DEBUG, pubsub ? "+pubsub-link" : "+cmd-link", ri,
933 "%@");
934 }
935 }
936
937 void sentinelDisconnectCallback(const redisAsyncContext *c, int status) {
938 sentinelDisconnectInstanceFromContext(c);
939 }
940
941 /* Create the async connections for the specified instance if the instance
942 * is disconnected. Note that the SRI_DISCONNECTED flag is set even if just
943 * one of the two links (commands and pub/sub) is missing. */
944 void sentinelReconnectInstance(sentinelRedisInstance *ri) {
945 if (!(ri->flags & SRI_DISCONNECTED)) return;
946
947 /* Commands connection. */
948 if (ri->cc == NULL) {
949 ri->cc = redisAsyncConnect(ri->addr->ip,ri->addr->port);
950 if (ri->cc->err) {
951 sentinelEvent(REDIS_DEBUG,"-cmd-link-reconnection",ri,"%@ #%s",
952 ri->cc->errstr);
953 sentinelKillLink(ri,ri->cc);
954 } else {
955 ri->cc_conn_time = mstime();
956 ri->cc->data = ri;
957 redisAeAttach(server.el,ri->cc);
958 redisAsyncSetConnectCallback(ri->cc,
959 sentinelLinkEstablishedCallback);
960 redisAsyncSetDisconnectCallback(ri->cc,
961 sentinelDisconnectCallback);
962 }
963 }
964 /* Pub / Sub */
965 if ((ri->flags & SRI_MASTER) && ri->pc == NULL) {
966 ri->pc = redisAsyncConnect(ri->addr->ip,ri->addr->port);
967 if (ri->pc->err) {
968 sentinelEvent(REDIS_DEBUG,"-pubsub-link-reconnection",ri,"%@ #%s",
969 ri->pc->errstr);
970 sentinelKillLink(ri,ri->pc);
971 } else {
972 int retval;
973
974 ri->pc_conn_time = mstime();
975 ri->pc->data = ri;
976 redisAeAttach(server.el,ri->pc);
977 redisAsyncSetConnectCallback(ri->pc,
978 sentinelLinkEstablishedCallback);
979 redisAsyncSetDisconnectCallback(ri->pc,
980 sentinelDisconnectCallback);
981 /* Now we subscribe to the Sentinels "Hello" channel. */
982 retval = redisAsyncCommand(ri->pc,
983 sentinelReceiveHelloMessages, NULL, "SUBSCRIBE %s",
984 SENTINEL_HELLO_CHANNEL);
985 if (retval != REDIS_OK) {
986 /* If we can't subscribe, the Pub/Sub connection is useless
987 * and we can simply disconnect it and try again. */
988 sentinelKillLink(ri,ri->pc);
989 return;
990 }
991 }
992 }
993 /* Clear the DISCONNECTED flags only if we have both the connections
994 * (or just the commands connection if this is a slave or a
995 * sentinel instance). */
996 if (ri->cc && (ri->flags & (SRI_SLAVE|SRI_SENTINEL) || ri->pc))
997 ri->flags &= ~SRI_DISCONNECTED;
998 }
999
1000 /* ======================== Redis instances pinging ======================== */
1001
1002 /* Process the INFO output from masters. */
1003 void sentinelRefreshInstanceInfo(sentinelRedisInstance *ri, const char *info) {
1004 sds *lines;
1005 int numlines, j;
1006 int role = 0;
1007 int runid_changed = 0; /* true if runid changed. */
1008 int first_runid = 0; /* true if this is the first runid we receive. */
1009
1010 /* The following fields must be reset to a given value in the case they
1011 * are not found at all in the INFO output. */
1012 ri->master_link_down_time = 0;
1013
1014 /* Process line by line. */
1015 lines = sdssplitlen(info,strlen(info),"\r\n",2,&numlines);
1016 for (j = 0; j < numlines; j++) {
1017 sentinelRedisInstance *slave;
1018 sds l = lines[j];
1019
1020 /* run_id:<40 hex chars>*/
1021 if (sdslen(l) >= 47 && !memcmp(l,"run_id:",7)) {
1022 if (ri->runid == NULL) {
1023 ri->runid = sdsnewlen(l+7,40);
1024 first_runid = 1;
1025 } else {
1026 if (strncmp(ri->runid,l+7,40) != 0) {
1027 runid_changed = 1;
1028 sentinelEvent(REDIS_NOTICE,"+reboot",ri,"%@");
1029 sdsfree(ri->runid);
1030 ri->runid = sdsnewlen(l+7,40);
1031 }
1032 }
1033 }
1034
1035 /* slave0:<ip>,<port>,<state> */
1036 if ((ri->flags & SRI_MASTER) &&
1037 sdslen(l) >= 7 &&
1038 !memcmp(l,"slave",5) && isdigit(l[5]))
1039 {
1040 char *ip, *port, *end;
1041
1042 ip = strchr(l,':'); if (!ip) continue;
1043 ip++; /* Now ip points to start of ip address. */
1044 port = strchr(ip,','); if (!port) continue;
1045 *port = '\0'; /* nul term for easy access. */
1046 port++; /* Now port points to start of port number. */
1047 end = strchr(port,','); if (!end) continue;
1048 *end = '\0'; /* nul term for easy access. */
1049
1050 /* Check if we already have this slave into our table,
1051 * otherwise add it. */
1052 if (sentinelRedisInstanceLookupSlave(ri,ip,atoi(port)) == NULL) {
1053 if ((slave = createSentinelRedisInstance(NULL,SRI_SLAVE,ip,
1054 atoi(port), ri->quorum,ri)) != NULL)
1055 {
1056 sentinelEvent(REDIS_NOTICE,"+slave",slave,"%@");
1057 }
1058 }
1059 }
1060
1061 /* master_link_down_since_seconds:<seconds> */
1062 if (sdslen(l) >= 32 &&
1063 !memcmp(l,"master_link_down_since_seconds",30))
1064 {
1065 ri->master_link_down_time = strtoll(l+31,NULL,10)*1000;
1066 }
1067
1068 /* role:<role> */
1069 if (!memcmp(l,"role:master",11)) role = SRI_MASTER;
1070 else if (!memcmp(l,"role:slave",10)) role = SRI_SLAVE;
1071
1072 if (role == SRI_SLAVE) {
1073 /* master_host:<host> */
1074 if (sdslen(l) >= 12 && !memcmp(l,"master_host:",12)) {
1075 sdsfree(ri->slave_master_host);
1076 ri->slave_master_host = sdsnew(l+12);
1077 }
1078
1079 /* master_port:<port> */
1080 if (sdslen(l) >= 12 && !memcmp(l,"master_port:",12))
1081 ri->slave_master_port = atoi(l+12);
1082
1083 /* master_link_status:<status> */
1084 if (sdslen(l) >= 19 && !memcmp(l,"master_link_status:",19)) {
1085 ri->slave_master_link_status =
1086 (strcasecmp(l+19,"up") == 0) ?
1087 SENTINEL_MASTER_LINK_STATUS_UP :
1088 SENTINEL_MASTER_LINK_STATUS_DOWN;
1089 }
1090 }
1091 }
1092 ri->info_refresh = mstime();
1093 sdsfreesplitres(lines,numlines);
1094
1095 if (sentinel.tilt) return;
1096
1097 /* Act if a master turned into a slave. */
1098 if ((ri->flags & SRI_MASTER) && role == SRI_SLAVE) {
1099 if (first_runid && ri->slave_master_host) {
1100 /* If it is the first time we receive INFO from it, but it's
1101 * a slave while it was configured as a master, we want to monitor
1102 * its master instead. */
1103 sentinelEvent(REDIS_WARNING,"+redirect-to-master",ri,
1104 "%s %s %d %s %d",
1105 ri->name, ri->addr->ip, ri->addr->port,
1106 ri->slave_master_host, ri->slave_master_port);
1107 sentinelResetMasterAndChangeAddress(ri,ri->slave_master_host,
1108 ri->slave_master_port);
1109 return;
1110 }
1111 }
1112
1113 /* Act if a slave turned into a master. */
1114 if ((ri->flags & SRI_SLAVE) && role == SRI_MASTER) {
1115 if (!(ri->master->flags & SRI_FAILOVER_IN_PROGRESS) &&
1116 (runid_changed || first_runid))
1117 {
1118 /* If a slave turned into a master, but at the same time the
1119 * runid has changed, or it is simply the first time we see and
1120 * INFO output from this instance, this is a reboot with a wrong
1121 * configuration.
1122 *
1123 * Log the event and remove the slave. */
1124 int retval;
1125
1126 sentinelEvent(REDIS_WARNING,"-slave-restart-as-master",ri,"%@ #removing it from the attached slaves");
1127 retval = dictDelete(ri->master->slaves,ri->name);
1128 redisAssert(retval == REDIS_OK);
1129 return;
1130 } else if (ri->flags & SRI_PROMOTED) {
1131 /* If this is a promoted slave we can change state to the
1132 * failover state machine. */
1133 if (ri->master &&
1134 (ri->master->flags & SRI_FAILOVER_IN_PROGRESS) &&
1135 (ri->master->flags & SRI_I_AM_THE_LEADER) &&
1136 (ri->master->failover_state ==
1137 SENTINEL_FAILOVER_STATE_WAIT_PROMOTION))
1138 {
1139 ri->master->failover_state = SENTINEL_FAILOVER_STATE_RECONF_SLAVES;
1140 ri->master->failover_state_change_time = mstime();
1141 sentinelEvent(REDIS_WARNING,"+promoted-slave",ri,"%@");
1142 sentinelEvent(REDIS_WARNING,"+failover-state-reconf-slaves",
1143 ri->master,"%@");
1144 }
1145 } else {
1146 /* Otherwise we interpret this as the start of the failover. */
1147 if (ri->master &&
1148 (ri->master->flags & SRI_FAILOVER_IN_PROGRESS) == 0)
1149 {
1150 ri->master->flags |= SRI_FAILOVER_IN_PROGRESS;
1151 sentinelEvent(REDIS_WARNING,"failover-detected",ri->master,"%@");
1152 ri->master->failover_state = SENTINEL_FAILOVER_STATE_DETECT_END;
1153 ri->master->failover_state_change_time = mstime();
1154 ri->master->promoted_slave = ri;
1155 ri->flags |= SRI_PROMOTED;
1156 /* We are an observer, so we can only assume that the leader
1157 * is reconfiguring the slave instances. For this reason we
1158 * set all the instances as RECONF_SENT waiting for progresses
1159 * on this side. */
1160 sentinelAddFlagsToDictOfRedisInstances(ri->master->slaves,
1161 SRI_RECONF_SENT);
1162 }
1163 }
1164 }
1165
1166 /* Detect if the slave that is in the process of being reconfigured
1167 * changed state. */
1168 if ((ri->flags & SRI_SLAVE) && role == SRI_SLAVE &&
1169 (ri->flags & (SRI_RECONF_SENT|SRI_RECONF_INPROG)))
1170 {
1171 /* SRI_RECONF_SENT -> SRI_RECONF_INPROG. */
1172 if ((ri->flags & SRI_RECONF_SENT) &&
1173 ri->slave_master_host &&
1174 strcmp(ri->slave_master_host,
1175 ri->master->promoted_slave->addr->ip) == 0 &&
1176 ri->slave_master_port == ri->master->promoted_slave->addr->port)
1177 {
1178 ri->flags &= ~SRI_RECONF_SENT;
1179 ri->flags |= SRI_RECONF_INPROG;
1180 sentinelEvent(REDIS_NOTICE,"+slave-reconf-inprog",ri,"%@");
1181 }
1182
1183 /* SRI_RECONF_INPROG -> SRI_RECONF_DONE */
1184 if ((ri->flags & SRI_RECONF_INPROG) &&
1185 ri->slave_master_link_status == SENTINEL_MASTER_LINK_STATUS_UP)
1186 {
1187 ri->flags &= ~SRI_RECONF_INPROG;
1188 ri->flags |= SRI_RECONF_DONE;
1189 sentinelEvent(REDIS_NOTICE,"+slave-reconf-done",ri,"%@");
1190 /* If we are moving forward (a new slave is now configured)
1191 * we update the change_time as we are conceptually passing
1192 * to the next slave. */
1193 ri->failover_state_change_time = mstime();
1194 }
1195 }
1196 }
1197
1198 void sentinelInfoReplyCallback(redisAsyncContext *c, void *reply, void *privdata) {
1199 sentinelRedisInstance *ri = c->data;
1200 redisReply *r;
1201
1202 if (ri) ri->pending_commands--;
1203 if (!reply || !ri) return;
1204 r = reply;
1205
1206 if (r->type == REDIS_REPLY_STRING) {
1207 sentinelRefreshInstanceInfo(ri,r->str);
1208 }
1209 }
1210
1211 /* Just discard the reply. We use this when we are not monitoring the return
1212 * value of the command but its effects directly. */
1213 void sentinelDiscardReplyCallback(redisAsyncContext *c, void *reply, void *privdata) {
1214 sentinelRedisInstance *ri = c->data;
1215
1216 if (ri) ri->pending_commands--;
1217 }
1218
1219 void sentinelPingReplyCallback(redisAsyncContext *c, void *reply, void *privdata) {
1220 sentinelRedisInstance *ri = c->data;
1221 redisReply *r;
1222
1223 if (ri) ri->pending_commands--;
1224 if (!reply || !ri) return;
1225 r = reply;
1226
1227 if (r->type == REDIS_REPLY_STATUS ||
1228 r->type == REDIS_REPLY_ERROR) {
1229 /* Update the "instance available" field only if this is an
1230 * acceptable reply. */
1231 if (strncmp(r->str,"PONG",4) == 0 ||
1232 strncmp(r->str,"LOADING",7) == 0 ||
1233 strncmp(r->str,"MASTERDOWN",10) == 0)
1234 {
1235 ri->last_avail_time = mstime();
1236 }
1237 }
1238 ri->last_pong_time = mstime();
1239 }
1240
1241 /* This is called when we get the reply about the PUBLISH command we send
1242 * to the master to advertise this sentinel. */
1243 void sentinelPublishReplyCallback(redisAsyncContext *c, void *reply, void *privdata) {
1244 sentinelRedisInstance *ri = c->data;
1245 redisReply *r;
1246
1247 if (ri) ri->pending_commands--;
1248 if (!reply || !ri) return;
1249 r = reply;
1250
1251 /* Only update pub_time if we actually published our message. Otherwise
1252 * we'll retry against in 100 milliseconds. */
1253 if (r->type != REDIS_REPLY_ERROR)
1254 ri->last_pub_time = mstime();
1255 }
1256
1257 /* This is our Pub/Sub callback for the Hello channel. It's useful in order
1258 * to discover other sentinels attached at the same master. */
1259 void sentinelReceiveHelloMessages(redisAsyncContext *c, void *reply, void *privdata) {
1260 sentinelRedisInstance *ri = c->data;
1261 redisReply *r;
1262
1263 if (!reply || !ri) return;
1264 r = reply;
1265
1266 /* Update the last activity in the pubsub channel. Note that since we
1267 * receive our messages as well this timestamp can be used to detect
1268 * if the link is probably diconnected even if it seems otherwise. */
1269 ri->pc_last_activity = mstime();
1270
1271 /* Sanity check in the reply we expect, so that the code that follows
1272 * can avoid to check for details. */
1273 if (r->type != REDIS_REPLY_ARRAY ||
1274 r->elements != 3 ||
1275 r->element[0]->type != REDIS_REPLY_STRING ||
1276 r->element[1]->type != REDIS_REPLY_STRING ||
1277 r->element[2]->type != REDIS_REPLY_STRING ||
1278 strcmp(r->element[0]->str,"message") != 0) return;
1279
1280 /* We are not interested in meeting ourselves */
1281 if (strstr(r->element[2]->str,server.runid) != NULL) return;
1282
1283 {
1284 int numtokens, port, removed, canfailover;
1285 char **token = sdssplitlen(r->element[2]->str,
1286 r->element[2]->len,
1287 ":",1,&numtokens);
1288 sentinelRedisInstance *sentinel;
1289
1290 if (numtokens == 4) {
1291 /* First, try to see if we already have this sentinel. */
1292 port = atoi(token[1]);
1293 canfailover = atoi(token[3]);
1294 sentinel = getSentinelRedisInstanceByAddrAndRunID(
1295 ri->sentinels,token[0],port,token[2]);
1296
1297 if (!sentinel) {
1298 /* If not, remove all the sentinels that have the same runid
1299 * OR the same ip/port, because it's either a restart or a
1300 * network topology change. */
1301 removed = removeMatchingSentinelsFromMaster(ri,token[0],port,
1302 token[2]);
1303 if (removed) {
1304 sentinelEvent(REDIS_NOTICE,"-dup-sentinel",ri,
1305 "%@ #duplicate of %s:%d or %s",
1306 token[0],port,token[2]);
1307 }
1308
1309 /* Add the new sentinel. */
1310 sentinel = createSentinelRedisInstance(NULL,SRI_SENTINEL,
1311 token[0],port,ri->quorum,ri);
1312 if (sentinel) {
1313 sentinelEvent(REDIS_NOTICE,"+sentinel",sentinel,"%@");
1314 /* The runid is NULL after a new instance creation and
1315 * for Sentinels we don't have a later chance to fill it,
1316 * so do it now. */
1317 sentinel->runid = sdsnew(token[2]);
1318 }
1319 }
1320
1321 /* Update the state of the Sentinel. */
1322 if (sentinel) {
1323 sentinel->last_hello_time = mstime();
1324 if (canfailover)
1325 sentinel->flags |= SRI_CAN_FAILOVER;
1326 else
1327 sentinel->flags &= ~SRI_CAN_FAILOVER;
1328 }
1329 }
1330 sdsfreesplitres(token,numtokens);
1331 }
1332 }
1333
1334 void sentinelPingInstance(sentinelRedisInstance *ri) {
1335 mstime_t now = mstime();
1336 mstime_t info_period;
1337 int retval;
1338
1339 /* Return ASAP if we have already a PING or INFO already pending, or
1340 * in the case the instance is not properly connected. */
1341 if (ri->flags & SRI_DISCONNECTED) return;
1342
1343 /* For INFO, PING, PUBLISH that are not critical commands to send we
1344 * also have a limit of SENTINEL_MAX_PENDING_COMMANDS. We don't
1345 * want to use a lot of memory just because a link is not working
1346 * properly (note that anyway there is a redundant protection about this,
1347 * that is, the link will be disconnected and reconnected if a long
1348 * timeout condition is detected. */
1349 if (ri->pending_commands >= SENTINEL_MAX_PENDING_COMMANDS) return;
1350
1351 /* If this is a slave of a master in O_DOWN condition we start sending
1352 * it INFO every second, instead of the usual SENTINEL_INFO_PERIOD
1353 * period. In this state we want to closely monitor slaves in case they
1354 * are turned into masters by another Sentinel, or by the sysadmin. */
1355 if ((ri->flags & SRI_SLAVE) &&
1356 (ri->master->flags & (SRI_O_DOWN|SRI_FAILOVER_IN_PROGRESS))) {
1357 info_period = 1000;
1358 } else {
1359 info_period = SENTINEL_INFO_PERIOD;
1360 }
1361
1362 if ((ri->flags & SRI_SENTINEL) == 0 &&
1363 (ri->info_refresh == 0 ||
1364 (now - ri->info_refresh) > info_period))
1365 {
1366 /* Send INFO to masters and slaves, not sentinels. */
1367 retval = redisAsyncCommand(ri->cc,
1368 sentinelInfoReplyCallback, NULL, "INFO");
1369 if (retval != REDIS_OK) return;
1370 ri->pending_commands++;
1371 } else if ((now - ri->last_pong_time) > SENTINEL_PING_PERIOD) {
1372 /* Send PING to all the three kinds of instances. */
1373 retval = redisAsyncCommand(ri->cc,
1374 sentinelPingReplyCallback, NULL, "PING");
1375 if (retval != REDIS_OK) return;
1376 ri->pending_commands++;
1377 } else if ((ri->flags & SRI_MASTER) &&
1378 (now - ri->last_pub_time) > SENTINEL_PUBLISH_PERIOD)
1379 {
1380 /* PUBLISH hello messages only to masters. */
1381 struct sockaddr_in sa;
1382 socklen_t salen = sizeof(sa);
1383
1384 if (getsockname(ri->cc->c.fd,(struct sockaddr*)&sa,&salen) != -1) {
1385 char myaddr[128];
1386
1387 snprintf(myaddr,sizeof(myaddr),"%s:%d:%s:%d",
1388 inet_ntoa(sa.sin_addr), server.port, server.runid,
1389 (ri->flags & SRI_CAN_FAILOVER) != 0);
1390 retval = redisAsyncCommand(ri->cc,
1391 sentinelPublishReplyCallback, NULL, "PUBLISH %s %s",
1392 SENTINEL_HELLO_CHANNEL,myaddr);
1393 if (retval != REDIS_OK) return;
1394 ri->pending_commands++;
1395 }
1396 }
1397 }
1398
1399 /* =========================== SENTINEL command ============================= */
1400
1401 const char *sentinelFailoverStateStr(int state) {
1402 switch(state) {
1403 case SENTINEL_FAILOVER_STATE_NONE: return "none";
1404 case SENTINEL_FAILOVER_STATE_WAIT_START: return "wait_start";
1405 case SENTINEL_FAILOVER_STATE_SELECT_SLAVE: return "select_slave";
1406 case SENTINEL_FAILOVER_STATE_SEND_SLAVEOF_NOONE: return "send_slaveof_noone";
1407 case SENTINEL_FAILOVER_STATE_WAIT_PROMOTION: return "wait_promotion";
1408 case SENTINEL_FAILOVER_STATE_RECONF_SLAVES: return "reconf_slaves";
1409 case SENTINEL_FAILOVER_STATE_ALERT_CLIENTS: return "alert_clients";
1410 case SENTINEL_FAILOVER_STATE_DETECT_END: return "detect_end";
1411 case SENTINEL_FAILOVER_STATE_UPDATE_CONFIG: return "update_config";
1412 default: return "unknown";
1413 }
1414 }
1415
1416 /* Redis instance to Redis protocol representation. */
1417 void addReplySentinelRedisInstance(redisClient *c, sentinelRedisInstance *ri) {
1418 char *flags = sdsempty();
1419 void *mbl;
1420 int fields = 0;
1421
1422 mbl = addDeferredMultiBulkLength(c);
1423
1424 addReplyBulkCString(c,"name");
1425 addReplyBulkCString(c,ri->name);
1426 fields++;
1427
1428 addReplyBulkCString(c,"ip");
1429 addReplyBulkCString(c,ri->addr->ip);
1430 fields++;
1431
1432 addReplyBulkCString(c,"port");
1433 addReplyBulkLongLong(c,ri->addr->port);
1434 fields++;
1435
1436 addReplyBulkCString(c,"runid");
1437 addReplyBulkCString(c,ri->runid ? ri->runid : "");
1438 fields++;
1439
1440 addReplyBulkCString(c,"flags");
1441 if (ri->flags & SRI_S_DOWN) flags = sdscat(flags,"s_down,");
1442 if (ri->flags & SRI_O_DOWN) flags = sdscat(flags,"o_down,");
1443 if (ri->flags & SRI_MASTER) flags = sdscat(flags,"master,");
1444 if (ri->flags & SRI_SLAVE) flags = sdscat(flags,"slave,");
1445 if (ri->flags & SRI_SENTINEL) flags = sdscat(flags,"sentinel,");
1446 if (ri->flags & SRI_DISCONNECTED) flags = sdscat(flags,"disconnected,");
1447 if (ri->flags & SRI_MASTER_DOWN) flags = sdscat(flags,"master_down,");
1448 if (ri->flags & SRI_FAILOVER_IN_PROGRESS)
1449 flags = sdscat(flags,"failover_in_progress,");
1450 if (ri->flags & SRI_I_AM_THE_LEADER)
1451 flags = sdscat(flags,"i_am_the_leader,");
1452 if (ri->flags & SRI_PROMOTED) flags = sdscat(flags,"promoted,");
1453 if (ri->flags & SRI_RECONF_SENT) flags = sdscat(flags,"reconf_sent,");
1454 if (ri->flags & SRI_RECONF_INPROG) flags = sdscat(flags,"reconf_inprog,");
1455 if (ri->flags & SRI_RECONF_DONE) flags = sdscat(flags,"reconf_done,");
1456
1457 if (sdslen(flags) != 0) flags = sdsrange(flags,0,-2); /* remove last "," */
1458 addReplyBulkCString(c,flags);
1459 sdsfree(flags);
1460 fields++;
1461
1462 addReplyBulkCString(c,"pending-commands");
1463 addReplyBulkLongLong(c,ri->pending_commands);
1464 fields++;
1465
1466 if (ri->flags & SRI_FAILOVER_IN_PROGRESS) {
1467 addReplyBulkCString(c,"failover-state");
1468 addReplyBulkCString(c,(char*)sentinelFailoverStateStr(ri->failover_state));
1469 fields++;
1470 }
1471
1472 addReplyBulkCString(c,"last-ok-ping-reply");
1473 addReplyBulkLongLong(c,mstime() - ri->last_avail_time);
1474 fields++;
1475
1476 addReplyBulkCString(c,"last-ping-reply");
1477 addReplyBulkLongLong(c,mstime() - ri->last_pong_time);
1478 fields++;
1479
1480 if (ri->flags & SRI_S_DOWN) {
1481 addReplyBulkCString(c,"s-down-time");
1482 addReplyBulkLongLong(c,mstime()-ri->s_down_since_time);
1483 fields++;
1484 }
1485
1486 if (ri->flags & SRI_O_DOWN) {
1487 addReplyBulkCString(c,"o-down-time");
1488 addReplyBulkLongLong(c,mstime()-ri->o_down_since_time);
1489 fields++;
1490 }
1491
1492 /* Masters and Slaves */
1493 if (ri->flags & (SRI_MASTER|SRI_SLAVE)) {
1494 addReplyBulkCString(c,"info-refresh");
1495 addReplyBulkLongLong(c,mstime() - ri->info_refresh);
1496 fields++;
1497 }
1498
1499 /* Only masters */
1500 if (ri->flags & SRI_MASTER) {
1501 addReplyBulkCString(c,"num-slaves");
1502 addReplyBulkLongLong(c,dictSize(ri->slaves));
1503 fields++;
1504
1505 addReplyBulkCString(c,"num-other-sentinels");
1506 addReplyBulkLongLong(c,dictSize(ri->sentinels));
1507 fields++;
1508
1509 addReplyBulkCString(c,"quorum");
1510 addReplyBulkLongLong(c,ri->quorum);
1511 fields++;
1512 }
1513
1514 /* Only slaves */
1515 if (ri->flags & SRI_SLAVE) {
1516 addReplyBulkCString(c,"master-link-down-time");
1517 addReplyBulkLongLong(c,ri->master_link_down_time);
1518 fields++;
1519
1520 addReplyBulkCString(c,"master-link-status");
1521 addReplyBulkCString(c,
1522 (ri->slave_master_link_status == SENTINEL_MASTER_LINK_STATUS_UP) ?
1523 "ok" : "err");
1524 fields++;
1525
1526 addReplyBulkCString(c,"master-host");
1527 addReplyBulkCString(c,
1528 ri->slave_master_host ? ri->slave_master_host : "?");
1529 fields++;
1530
1531 addReplyBulkCString(c,"master-port");
1532 addReplyBulkLongLong(c,ri->slave_master_port);
1533 fields++;
1534 }
1535
1536 /* Only sentinels */
1537 if (ri->flags & SRI_SENTINEL) {
1538 addReplyBulkCString(c,"last-hello-message");
1539 addReplyBulkLongLong(c,mstime() - ri->last_hello_time);
1540 fields++;
1541
1542 addReplyBulkCString(c,"can-failover-its-master");
1543 addReplyBulkLongLong(c,(ri->flags & SRI_CAN_FAILOVER) != 0);
1544 fields++;
1545
1546 if (ri->flags & SRI_MASTER_DOWN) {
1547 addReplyBulkCString(c,"subjective-leader");
1548 addReplyBulkCString(c,ri->leader ? ri->leader : "?");
1549 fields++;
1550 }
1551 }
1552
1553 setDeferredMultiBulkLength(c,mbl,fields*2);
1554 }
1555
1556 /* Output a number of instances contanined inside a dictionary as
1557 * Redis protocol. */
1558 void addReplyDictOfRedisInstances(redisClient *c, dict *instances) {
1559 dictIterator *di;
1560 dictEntry *de;
1561
1562 di = dictGetIterator(instances);
1563 addReplyMultiBulkLen(c,dictSize(instances));
1564 while((de = dictNext(di)) != NULL) {
1565 sentinelRedisInstance *ri = dictGetVal(de);
1566
1567 addReplySentinelRedisInstance(c,ri);
1568 }
1569 dictReleaseIterator(di);
1570 }
1571
1572 /* Lookup the named master into sentinel.masters.
1573 * If the master is not found reply to the client with an error and returns
1574 * NULL. */
1575 sentinelRedisInstance *sentinelGetMasterByNameOrReplyError(redisClient *c,
1576 robj *name)
1577 {
1578 sentinelRedisInstance *ri;
1579
1580 ri = dictFetchValue(sentinel.masters,c->argv[2]->ptr);
1581 if (!ri) {
1582 addReplyError(c,"No such master with that name");
1583 return NULL;
1584 }
1585 return ri;
1586 }
1587
1588 void sentinelCommand(redisClient *c) {
1589 if (!strcasecmp(c->argv[1]->ptr,"masters")) {
1590 /* SENTINEL MASTERS */
1591 if (c->argc != 2) goto numargserr;
1592
1593 addReplyDictOfRedisInstances(c,sentinel.masters);
1594 } else if (!strcasecmp(c->argv[1]->ptr,"slaves")) {
1595 /* SENTINEL SLAVES <master-name> */
1596 sentinelRedisInstance *ri;
1597
1598 if (c->argc != 3) goto numargserr;
1599 if ((ri = sentinelGetMasterByNameOrReplyError(c,c->argv[2])) == NULL)
1600 return;
1601 addReplyDictOfRedisInstances(c,ri->slaves);
1602 } else if (!strcasecmp(c->argv[1]->ptr,"sentinels")) {
1603 /* SENTINEL SENTINELS <master-name> */
1604 sentinelRedisInstance *ri;
1605
1606 if (c->argc != 3) goto numargserr;
1607 if ((ri = sentinelGetMasterByNameOrReplyError(c,c->argv[2])) == NULL)
1608 return;
1609 addReplyDictOfRedisInstances(c,ri->sentinels);
1610 } else if (!strcasecmp(c->argv[1]->ptr,"is-master-down-by-addr")) {
1611 /* SENTINEL IS-MASTER-DOWN-BY-ADDR <ip> <port> */
1612 sentinelRedisInstance *ri;
1613 char *leader = NULL;
1614 long port;
1615 int isdown = 0;
1616
1617 if (c->argc != 4) goto numargserr;
1618 if (getLongFromObjectOrReply(c,c->argv[3],&port,NULL) != REDIS_OK)
1619 return;
1620 ri = getSentinelRedisInstanceByAddrAndRunID(sentinel.masters,
1621 c->argv[2]->ptr,port,NULL);
1622
1623 /* It exists? Is actually a master? Is subjectively down? It's down.
1624 * Note: if we are in tilt mode we always reply with "0". */
1625 if (!sentinel.tilt && ri && (ri->flags & SRI_S_DOWN) &&
1626 (ri->flags & SRI_MASTER))
1627 isdown = 1;
1628 if (ri) leader = sentinelGetSubjectiveLeader(ri);
1629
1630 /* Reply with a two-elements multi-bulk reply: down state, leader. */
1631 addReplyMultiBulkLen(c,2);
1632 addReply(c, isdown ? shared.cone : shared.czero);
1633 addReplyBulkCString(c, leader ? leader : "?");
1634 if (leader) sdsfree(leader);
1635 } else if (!strcasecmp(c->argv[1]->ptr,"reset")) {
1636 /* SENTINEL RESET <pattern> */
1637 if (c->argc != 3) goto numargserr;
1638 addReplyLongLong(c,sentinelResetMastersByPattern(c->argv[2]->ptr,SENTINEL_GENERATE_EVENT));
1639 } else if (!strcasecmp(c->argv[1]->ptr,"get-master-addr-by-name")) {
1640 /* SENTINEL GET-MASTER-ADDR-BY-NAME <master-name> */
1641 sentinelRedisInstance *ri;
1642
1643 if (c->argc != 3) goto numargserr;
1644 ri = sentinelGetMasterByName(c->argv[2]->ptr);
1645 if (ri == NULL) {
1646 addReply(c,shared.nullmultibulk);
1647 } else {
1648 sentinelAddr *addr = ri->addr;
1649
1650 if ((ri->flags & SRI_FAILOVER_IN_PROGRESS) && ri->promoted_slave)
1651 addr = ri->promoted_slave->addr;
1652 addReplyMultiBulkLen(c,2);
1653 addReplyBulkCString(c,addr->ip);
1654 addReplyBulkLongLong(c,addr->port);
1655 }
1656 } else {
1657 addReplyErrorFormat(c,"Unknown sentinel subcommand '%s'",
1658 (char*)c->argv[1]->ptr);
1659 }
1660 return;
1661
1662 numargserr:
1663 addReplyErrorFormat(c,"Wrong number of commands for 'sentinel %s'",
1664 (char*)c->argv[1]->ptr);
1665 }
1666
1667 /* ===================== SENTINEL availability checks ======================= */
1668
1669 /* Is this instance down from our point of view? */
1670 void sentinelCheckSubjectivelyDown(sentinelRedisInstance *ri) {
1671 mstime_t elapsed = mstime() - ri->last_avail_time;
1672
1673 /* Check if we are in need for a reconnection of one of the
1674 * links, because we are detecting low activity.
1675 *
1676 * 1) Check if the command link seems connected, was connected not less
1677 * than SENTINEL_MIN_LINK_RECONNECT_PERIOD, but still we have an
1678 * idle time that is greater than down_after_period / 2 seconds. */
1679 if (ri->cc &&
1680 (mstime() - ri->cc_conn_time) > SENTINEL_MIN_LINK_RECONNECT_PERIOD &&
1681 (mstime() - ri->last_pong_time) > (ri->down_after_period/2))
1682 {
1683 sentinelKillLink(ri,ri->cc);
1684 }
1685
1686 /* 2) Check if the pubsub link seems connected, was connected not less
1687 * than SENTINEL_MIN_LINK_RECONNECT_PERIOD, but still we have no
1688 * activity in the Pub/Sub channel for more than
1689 * SENTINEL_PUBLISH_PERIOD * 3.
1690 */
1691 if (ri->pc &&
1692 (mstime() - ri->pc_conn_time) > SENTINEL_MIN_LINK_RECONNECT_PERIOD &&
1693 (mstime() - ri->pc_last_activity) > (SENTINEL_PUBLISH_PERIOD*3))
1694 {
1695 sentinelKillLink(ri,ri->pc);
1696 }
1697
1698 /* Update the subjectively down flag. */
1699 if (elapsed > ri->down_after_period) {
1700 /* Is subjectively down */
1701 if ((ri->flags & SRI_S_DOWN) == 0) {
1702 sentinelEvent(REDIS_WARNING,"+sdown",ri,"%@");
1703 ri->s_down_since_time = mstime();
1704 ri->flags |= SRI_S_DOWN;
1705 }
1706 } else {
1707 /* Is subjectively up */
1708 if (ri->flags & SRI_S_DOWN) {
1709 sentinelEvent(REDIS_WARNING,"-sdown",ri,"%@");
1710 ri->flags &= ~SRI_S_DOWN;
1711 }
1712 }
1713 }
1714
1715 /* Is this instance down accordingly to the configured quorum? */
1716 void sentinelCheckObjectivelyDown(sentinelRedisInstance *master) {
1717 dictIterator *di;
1718 dictEntry *de;
1719 int quorum = 0, odown = 0;
1720
1721 if (master->flags & SRI_S_DOWN) {
1722 /* Is down for enough sentinels? */
1723 quorum = 1; /* the current sentinel. */
1724 /* Count all the other sentinels. */
1725 di = dictGetIterator(master->sentinels);
1726 while((de = dictNext(di)) != NULL) {
1727 sentinelRedisInstance *ri = dictGetVal(de);
1728
1729 if (ri->flags & SRI_MASTER_DOWN) quorum++;
1730 }
1731 dictReleaseIterator(di);
1732 if (quorum >= master->quorum) odown = 1;
1733 }
1734
1735 /* Set the flag accordingly to the outcome. */
1736 if (odown) {
1737 if ((master->flags & SRI_O_DOWN) == 0) {
1738 sentinelEvent(REDIS_WARNING,"+odown",master,"%@ #quorum %d/%d",
1739 quorum, master->quorum);
1740 master->flags |= SRI_O_DOWN;
1741 master->o_down_since_time = mstime();
1742 }
1743 } else {
1744 if (master->flags & SRI_O_DOWN) {
1745 sentinelEvent(REDIS_WARNING,"-odown",master,"%@");
1746 master->flags &= ~SRI_O_DOWN;
1747 }
1748 }
1749 }
1750
1751 /* Receive the SENTINEL is-master-down-by-addr reply, see the
1752 * sentinelAskMasterStateToOtherSentinels() function for more information. */
1753 void sentinelReceiveIsMasterDownReply(redisAsyncContext *c, void *reply, void *privdata) {
1754 sentinelRedisInstance *ri = c->data;
1755 redisReply *r;
1756
1757 if (ri) ri->pending_commands--;
1758 if (!reply || !ri) return;
1759 r = reply;
1760
1761 /* Ignore every error or unexpected reply.
1762 * Note that if the command returns an error for any reason we'll
1763 * end clearing the SRI_MASTER_DOWN flag for timeout anyway. */
1764 if (r->type == REDIS_REPLY_ARRAY && r->elements == 2 &&
1765 r->element[0]->type == REDIS_REPLY_INTEGER &&
1766 r->element[1]->type == REDIS_REPLY_STRING)
1767 {
1768 ri->last_master_down_reply_time = mstime();
1769 if (r->element[0]->integer == 1) {
1770 ri->flags |= SRI_MASTER_DOWN;
1771 } else {
1772 ri->flags &= ~SRI_MASTER_DOWN;
1773 }
1774 sdsfree(ri->leader);
1775 ri->leader = sdsnew(r->element[1]->str);
1776 }
1777 }
1778
1779 /* If we think (subjectively) the master is down, we start sending
1780 * SENTINEL IS-MASTER-DOWN-BY-ADDR requests to other sentinels
1781 * in order to get the replies that allow to reach the quorum and
1782 * possibly also mark the master as objectively down. */
1783 void sentinelAskMasterStateToOtherSentinels(sentinelRedisInstance *master) {
1784 dictIterator *di;
1785 dictEntry *de;
1786
1787 di = dictGetIterator(master->sentinels);
1788 while((de = dictNext(di)) != NULL) {
1789 sentinelRedisInstance *ri = dictGetVal(de);
1790 mstime_t elapsed = mstime() - ri->last_master_down_reply_time;
1791 char port[32];
1792 int retval;
1793
1794 /* If the master state from other sentinel is too old, we clear it. */
1795 if (elapsed > SENTINEL_INFO_VALIDITY_TIME) {
1796 ri->flags &= ~SRI_MASTER_DOWN;
1797 sdsfree(ri->leader);
1798 ri->leader = NULL;
1799 }
1800
1801 /* Only ask if master is down to other sentinels if:
1802 *
1803 * 1) We believe it is down, or there is a failover in progress.
1804 * 2) Sentinel is connected.
1805 * 3) We did not received the info within SENTINEL_ASK_PERIOD ms. */
1806 if ((master->flags & (SRI_S_DOWN|SRI_FAILOVER_IN_PROGRESS)) == 0)
1807 continue;
1808 if (ri->flags & SRI_DISCONNECTED) continue;
1809 if (mstime() - ri->last_master_down_reply_time < SENTINEL_ASK_PERIOD)
1810 continue;
1811
1812 /* Ask */
1813 ll2string(port,sizeof(port),master->addr->port);
1814 retval = redisAsyncCommand(ri->cc,
1815 sentinelReceiveIsMasterDownReply, NULL,
1816 "SENTINEL is-master-down-by-addr %s %s",
1817 master->addr->ip, port);
1818 if (retval == REDIS_OK) ri->pending_commands++;
1819 }
1820 dictReleaseIterator(di);
1821 }
1822
1823 /* =============================== FAILOVER ================================= */
1824
1825 /* Given a master get the "subjective leader", that is, among all the sentinels
1826 * with given characteristics, the one with the lexicographically smaller
1827 * runid. The characteristics required are:
1828 *
1829 * 1) Has SRI_CAN_FAILOVER flag.
1830 * 2) Is not disconnected.
1831 * 3) Recently answered to our ping (no longer than
1832 * SENTINEL_INFO_VALIDITY_TIME milliseconds ago).
1833 *
1834 * The function returns a pointer to an sds string representing the runid of the
1835 * leader sentinel instance (from our point of view). Otherwise NULL is
1836 * returned if there are no suitable sentinels.
1837 */
1838
1839 int compareRunID(const void *a, const void *b) {
1840 char **aptrptr = (char**)a, **bptrptr = (char**)b;
1841 return strcasecmp(*aptrptr, *bptrptr);
1842 }
1843
1844 char *sentinelGetSubjectiveLeader(sentinelRedisInstance *master) {
1845 dictIterator *di;
1846 dictEntry *de;
1847 char **instance =
1848 zmalloc(sizeof(char*)*(dictSize(master->sentinels)+1));
1849 int instances = 0;
1850 char *leader = NULL;
1851
1852 if (master->flags & SRI_CAN_FAILOVER) {
1853 /* Add myself if I'm a Sentinel that can failover this master. */
1854 instance[instances++] = server.runid;
1855 }
1856
1857 di = dictGetIterator(master->sentinels);
1858 while((de = dictNext(di)) != NULL) {
1859 sentinelRedisInstance *ri = dictGetVal(de);
1860 mstime_t lag = mstime() - ri->last_avail_time;
1861
1862 if (lag > SENTINEL_INFO_VALIDITY_TIME ||
1863 !(ri->flags & SRI_CAN_FAILOVER) ||
1864 (ri->flags & SRI_DISCONNECTED) ||
1865 ri->runid == NULL)
1866 continue;
1867 instance[instances++] = ri->runid;
1868 }
1869 dictReleaseIterator(di);
1870
1871 /* If we have at least one instance passing our checks, order the array
1872 * by runid. */
1873 if (instances) {
1874 qsort(instance,instances,sizeof(char*),compareRunID);
1875 leader = sdsnew(instance[0]);
1876 }
1877 zfree(instance);
1878 return leader;
1879 }
1880
1881 struct sentinelLeader {
1882 char *runid;
1883 unsigned long votes;
1884 };
1885
1886 /* Helper function for sentinelGetObjectiveLeader, increment the counter
1887 * relative to the specified runid. */
1888 void sentinelObjectiveLeaderIncr(dict *counters, char *runid) {
1889 dictEntry *de = dictFind(counters,runid);
1890 uint64_t oldval;
1891
1892 if (de) {
1893 oldval = dictGetUnsignedIntegerVal(de);
1894 dictSetUnsignedIntegerVal(de,oldval+1);
1895 } else {
1896 de = dictAddRaw(counters,runid);
1897 redisAssert(de != NULL);
1898 dictSetUnsignedIntegerVal(de,1);
1899 }
1900 }
1901
1902 /* Scan all the Sentinels attached to this master to check what is the
1903 * most voted leader among Sentinels. */
1904 char *sentinelGetObjectiveLeader(sentinelRedisInstance *master) {
1905 dict *counters;
1906 dictIterator *di;
1907 dictEntry *de;
1908 unsigned int voters = 0, voters_quorum;
1909 char *myvote;
1910 char *winner = NULL;
1911
1912 redisAssert(master->flags & (SRI_O_DOWN|SRI_FAILOVER_IN_PROGRESS));
1913 counters = dictCreate(&leaderVotesDictType,NULL);
1914
1915 /* Count my vote. */
1916 myvote = sentinelGetSubjectiveLeader(master);
1917 if (myvote) {
1918 sentinelObjectiveLeaderIncr(counters,myvote);
1919 voters++;
1920 }
1921
1922 /* Count other sentinels votes */
1923 di = dictGetIterator(master->sentinels);
1924 while((de = dictNext(di)) != NULL) {
1925 sentinelRedisInstance *ri = dictGetVal(de);
1926 if (ri->leader == NULL) continue;
1927 /* If the failover is not already in progress we are only interested
1928 * in Sentinels that believe the master is down. Otherwise the leader
1929 * selection is useful for the "failover-takedown" when the original
1930 * leader fails. In that case we consider all the voters. */
1931 if (!(master->flags & SRI_FAILOVER_IN_PROGRESS) &&
1932 !(ri->flags & SRI_MASTER_DOWN)) continue;
1933 sentinelObjectiveLeaderIncr(counters,ri->leader);
1934 voters++;
1935 }
1936 dictReleaseIterator(di);
1937 voters_quorum = voters/2+1;
1938
1939 /* Check what's the winner. For the winner to win, it needs two conditions:
1940 * 1) Absolute majority between voters (50% + 1).
1941 * 2) And anyway at least master->quorum votes. */
1942 {
1943 uint64_t max_votes = 0; /* Max votes so far. */
1944
1945 di = dictGetIterator(counters);
1946 while((de = dictNext(di)) != NULL) {
1947 uint64_t votes = dictGetUnsignedIntegerVal(de);
1948
1949 if (max_votes < votes) {
1950 max_votes = votes;
1951 winner = dictGetKey(de);
1952 }
1953 }
1954 dictReleaseIterator(di);
1955 if (winner && (max_votes < voters_quorum || max_votes < master->quorum))
1956 winner = NULL;
1957 }
1958 winner = winner ? sdsnew(winner) : NULL;
1959 sdsfree(myvote);
1960 dictRelease(counters);
1961 return winner;
1962 }
1963
1964 /* This function checks if there are the conditions to start the failover,
1965 * that is:
1966 *
1967 * 1) Enough time has passed since O_DOWN.
1968 * 2) The master is marked as SRI_CAN_FAILOVER, so we can failover it.
1969 * 3) We are the objectively leader for this master.
1970 *
1971 * If the conditions are met we flag the master as SRI_FAILOVER_IN_PROGRESS
1972 * and SRI_I_AM_THE_LEADER.
1973 */
1974 void sentinelStartFailover(sentinelRedisInstance *master) {
1975 char *leader;
1976 int isleader;
1977
1978 /* We can't failover if the master is not in O_DOWN state or if
1979 * there is not already a failover in progress (to perform the
1980 * takedown if the leader died) or if this Sentinel is not allowed
1981 * to start a failover. */
1982 if (!(master->flags & SRI_CAN_FAILOVER) ||
1983 !(master->flags & (SRI_O_DOWN|SRI_FAILOVER_IN_PROGRESS))) return;
1984
1985 leader = sentinelGetObjectiveLeader(master);
1986 isleader = leader && strcasecmp(leader,server.runid) == 0;
1987 sdsfree(leader);
1988
1989 /* If I'm not the leader, I can't failover for sure. */
1990 if (!isleader) return;
1991
1992 /* If the failover is already in progress there are two options... */
1993 if (master->flags & SRI_FAILOVER_IN_PROGRESS) {
1994 if (master->flags & SRI_I_AM_THE_LEADER) {
1995 /* 1) I'm flagged as leader so I already started the failover.
1996 * Just return. */
1997 return;
1998 } else {
1999 mstime_t elapsed = mstime() - master->failover_state_change_time;
2000
2001 /* 2) I'm the new leader, but I'm not flagged as leader in the
2002 * master: I did not started the failover, but the original
2003 * leader has no longer the leadership.
2004 *
2005 * In this case if the failover appears to be lagging
2006 * for at least 25% of the configured failover timeout,
2007 * I can assume I can take control. Otherwise
2008 * it's better to return and wait more. */
2009 if (elapsed < (master->failover_timeout/4)) return;
2010 sentinelEvent(REDIS_WARNING,"+failover-takedown",master,"%@");
2011 /* We have already an elected slave if we are in
2012 * FAILOVER_IN_PROGRESS state, that is, the slave that we
2013 * observed turning into a master. */
2014 master->failover_state = SENTINEL_FAILOVER_STATE_RECONF_SLAVES;
2015 /* As an observer we flagged all the slaves as RECONF_SENT but
2016 * now we are in charge of actually sending the reconfiguration
2017 * command so let's clear this flag for all the instances. */
2018 sentinelDelFlagsToDictOfRedisInstances(master->slaves,
2019 SRI_RECONF_SENT);
2020 }
2021 } else {
2022 /* Brand new failover as SRI_FAILOVER_IN_PROGRESS was not set. */
2023 master->failover_state = SENTINEL_FAILOVER_STATE_WAIT_START;
2024 }
2025
2026 master->flags |= SRI_FAILOVER_IN_PROGRESS|SRI_I_AM_THE_LEADER;
2027 sentinelEvent(REDIS_WARNING,"+failover-triggered",master,"%@");
2028
2029 /* Pick a random delay if it's a fresh failover (WAIT_START), and not
2030 * a recovery of a failover started by another sentinel. */
2031 if (master->failover_state == SENTINEL_FAILOVER_STATE_WAIT_START) {
2032 master->failover_start_time = mstime() +
2033 SENTINEL_FAILOVER_FIXED_DELAY +
2034 (rand() % SENTINEL_FAILOVER_MAX_RANDOM_DELAY);
2035 sentinelEvent(REDIS_WARNING,"+failover-state-wait-start",master,
2036 "%@ #starting in %lld milliseconds",
2037 master->failover_start_time-mstime());
2038 }
2039 master->failover_state_change_time = mstime();
2040 }
2041
2042 /* Select a suitable slave to promote. The current algorithm only uses
2043 * the following parameters:
2044 *
2045 * 1) None of the following conditions: S_DOWN, O_DOWN, DISCONNECTED.
2046 * 2) last_avail_time more recent than SENTINEL_INFO_VALIDITY_TIME.
2047 * 3) info_refresh more recent than SENTINEL_INFO_VALIDITY_TIME.
2048 * 4) master_link_down_time no more than:
2049 * (now - master->s_down_since_time) + (master->down_after_period * 10).
2050 *
2051 * Among all the slaves matching the above conditions we select the slave
2052 * with lower slave_priority. If priority is the same we select the slave
2053 * with lexicographically smaller runid.
2054 *
2055 * The function returns the pointer to the selected slave, otherwise
2056 * NULL if no suitable slave was found.
2057 */
2058
2059 int compareSlavesForPromotion(const void *a, const void *b) {
2060 sentinelRedisInstance **sa = (sentinelRedisInstance **)a,
2061 **sb = (sentinelRedisInstance **)b;
2062 if ((*sa)->slave_priority != (*sb)->slave_priority)
2063 return (*sa)->slave_priority - (*sb)->slave_priority;
2064 return strcasecmp((*sa)->runid,(*sb)->runid);
2065 }
2066
2067 sentinelRedisInstance *sentinelSelectSlave(sentinelRedisInstance *master) {
2068 sentinelRedisInstance **instance =
2069 zmalloc(sizeof(instance[0])*dictSize(master->slaves));
2070 sentinelRedisInstance *selected = NULL;
2071 int instances = 0;
2072 dictIterator *di;
2073 dictEntry *de;
2074 mstime_t max_master_down_time;
2075
2076 max_master_down_time = (mstime() - master->s_down_since_time) +
2077 (master->down_after_period * 10);
2078
2079 di = dictGetIterator(master->slaves);
2080 while((de = dictNext(di)) != NULL) {
2081 sentinelRedisInstance *slave = dictGetVal(de);
2082 mstime_t info_validity_time = mstime()-SENTINEL_INFO_VALIDITY_TIME;
2083
2084 if (slave->flags & (SRI_S_DOWN|SRI_O_DOWN|SRI_DISCONNECTED)) continue;
2085 if (slave->last_avail_time < info_validity_time) continue;
2086 if (slave->info_refresh < info_validity_time) continue;
2087 if (slave->master_link_down_time > max_master_down_time) continue;
2088 instance[instances++] = slave;
2089 }
2090 dictReleaseIterator(di);
2091 if (instances) {
2092 qsort(instance,instances,sizeof(sentinelRedisInstance*),
2093 compareSlavesForPromotion);
2094 selected = instance[0];
2095 }
2096 zfree(instance);
2097 return selected;
2098 }
2099
2100 /* ---------------- Failover state machine implementation ------------------- */
2101 void sentinelFailoverWaitStart(sentinelRedisInstance *ri) {
2102 if (mstime() >= ri->failover_start_time) {
2103 ri->failover_state = SENTINEL_FAILOVER_STATE_SELECT_SLAVE;
2104 ri->failover_state_change_time = mstime();
2105 sentinelEvent(REDIS_WARNING,"+failover-state-select-slave",ri,"%@");
2106 }
2107 }
2108
2109 void sentinelFailoverSelectSlave(sentinelRedisInstance *ri) {
2110 sentinelRedisInstance *slave = sentinelSelectSlave(ri);
2111
2112 if (slave == NULL) {
2113 sentinelEvent(REDIS_WARNING,"-no-good-slave",ri,
2114 "%@ #retrying in %d seconds",
2115 (SENTINEL_FAILOVER_FIXED_DELAY+
2116 SENTINEL_FAILOVER_MAX_RANDOM_DELAY)/1000);
2117 ri->failover_state = SENTINEL_FAILOVER_STATE_WAIT_START;
2118 ri->failover_start_time = mstime() + SENTINEL_FAILOVER_FIXED_DELAY +
2119 SENTINEL_FAILOVER_MAX_RANDOM_DELAY;
2120 } else {
2121 sentinelEvent(REDIS_WARNING,"+selected-slave",slave,"%@");
2122 slave->flags |= SRI_PROMOTED;
2123 ri->promoted_slave = slave;
2124 ri->failover_state = SENTINEL_FAILOVER_STATE_SEND_SLAVEOF_NOONE;
2125 ri->failover_state_change_time = mstime();
2126 sentinelEvent(REDIS_NOTICE,"+failover-state-send-slaveof-noone",
2127 slave, "%@");
2128 }
2129 }
2130
2131 void sentinelFailoverSendSlaveOfNoOne(sentinelRedisInstance *ri) {
2132 int retval;
2133
2134 if (ri->promoted_slave->flags & SRI_DISCONNECTED) return;
2135
2136 /* Send SLAVEOF NO ONE command to turn the slave into a master.
2137 * We actually register a generic callback for this command as we don't
2138 * really care about the reply. We check if it worked indirectly observing
2139 * if INFO returns a different role (master instead of slave). */
2140 retval = redisAsyncCommand(ri->promoted_slave->cc,
2141 sentinelDiscardReplyCallback, NULL, "SLAVEOF NO ONE");
2142 if (retval != REDIS_OK) return;
2143 ri->promoted_slave->pending_commands++;
2144 sentinelEvent(REDIS_NOTICE, "+failover-state-wait-promotion",
2145 ri->promoted_slave,"%@");
2146 ri->failover_state = SENTINEL_FAILOVER_STATE_WAIT_PROMOTION;
2147 ri->failover_state_change_time = mstime();
2148 }
2149
2150 /* We actually wait for promotion indirectly checking with INFO when the
2151 * slave turns into a master. */
2152 void sentinelFailoverWaitPromotion(sentinelRedisInstance *ri) {
2153 mstime_t elapsed = mstime() - ri->failover_state_change_time;
2154
2155 if (elapsed >= SENTINEL_PROMOTION_RETRY_PERIOD) {
2156 sentinelEvent(REDIS_WARNING,"-promotion-timeout",ri->promoted_slave,
2157 "%@");
2158 sentinelEvent(REDIS_WARNING,"+failover-state-select-slave",ri,"%@");
2159 ri->failover_state = SENTINEL_FAILOVER_STATE_SELECT_SLAVE;
2160 ri->failover_state_change_time = mstime();
2161 ri->promoted_slave->flags &= ~SRI_PROMOTED;
2162 ri->promoted_slave = NULL;
2163 }
2164 }
2165
2166 void sentinelFailoverDetectEnd(sentinelRedisInstance *master) {
2167 int not_reconfigured = 0, timeout = 0;
2168 dictIterator *di;
2169 dictEntry *de;
2170 mstime_t elapsed = mstime() - master->failover_state_change_time;
2171
2172 /* We can't consider failover finished if the promoted slave is
2173 * not reachable. */
2174 if (master->promoted_slave == NULL ||
2175 master->promoted_slave->flags & SRI_S_DOWN) return;
2176
2177 /* The failover terminates once all the reachable slaves are properly
2178 * configured. */
2179 di = dictGetIterator(master->slaves);
2180 while((de = dictNext(di)) != NULL) {
2181 sentinelRedisInstance *slave = dictGetVal(de);
2182
2183 if (slave->flags & (SRI_PROMOTED|SRI_RECONF_DONE)) continue;
2184 if (slave->flags & SRI_S_DOWN) continue;
2185 not_reconfigured++;
2186 }
2187 dictReleaseIterator(di);
2188
2189 /* Force end of failover on timeout. */
2190 if (elapsed > master->failover_timeout) {
2191 not_reconfigured = 0;
2192 timeout = 1;
2193 sentinelEvent(REDIS_WARNING,"+failover-end-for-timeout",master,"%@");
2194 }
2195
2196 if (not_reconfigured == 0) {
2197 sentinelEvent(REDIS_WARNING,"+failover-end",master,"%@");
2198 master->failover_state = SENTINEL_FAILOVER_STATE_UPDATE_CONFIG;
2199 master->failover_state_change_time = mstime();
2200 }
2201
2202 /* If I'm the leader it is a good idea to send a best effort SLAVEOF
2203 * command to all the slaves still not reconfigured to replicate with
2204 * the new master. */
2205 if (timeout && (master->flags & SRI_I_AM_THE_LEADER)) {
2206 dictIterator *di;
2207 dictEntry *de;
2208 char master_port[32];
2209
2210 ll2string(master_port,sizeof(master_port),
2211 master->promoted_slave->addr->port);
2212
2213 di = dictGetIterator(master->slaves);
2214 while((de = dictNext(di)) != NULL) {
2215 sentinelRedisInstance *slave = dictGetVal(de);
2216 int retval;
2217
2218 if (slave->flags &
2219 (SRI_RECONF_DONE|SRI_RECONF_SENT|SRI_DISCONNECTED)) continue;
2220
2221 retval = redisAsyncCommand(slave->cc,
2222 sentinelDiscardReplyCallback, NULL, "SLAVEOF %s %s",
2223 master->promoted_slave->addr->ip,
2224 master_port);
2225 if (retval == REDIS_OK) {
2226 sentinelEvent(REDIS_NOTICE,"+slave-reconf-sent-be",slave,"%@");
2227 slave->flags |= SRI_RECONF_SENT;
2228 }
2229 }
2230 dictReleaseIterator(di);
2231 }
2232 }
2233
2234 /* Send SLAVE OF <new master address> to all the remaining slaves that
2235 * still don't appear to have the configuration updated. */
2236 void sentinelFailoverReconfNextSlave(sentinelRedisInstance *master) {
2237 dictIterator *di;
2238 dictEntry *de;
2239 int in_progress = 0;
2240
2241 di = dictGetIterator(master->slaves);
2242 while((de = dictNext(di)) != NULL) {
2243 sentinelRedisInstance *slave = dictGetVal(de);
2244
2245 if (slave->flags & (SRI_RECONF_SENT|SRI_RECONF_INPROG))
2246 in_progress++;
2247 }
2248 dictReleaseIterator(di);
2249
2250 di = dictGetIterator(master->slaves);
2251 while(in_progress < master->parallel_syncs &&
2252 (de = dictNext(di)) != NULL)
2253 {
2254 sentinelRedisInstance *slave = dictGetVal(de);
2255 int retval;
2256 char master_port[32];
2257
2258 /* Skip the promoted slave, and already configured slaves. */
2259 if (slave->flags & (SRI_PROMOTED|SRI_RECONF_DONE)) continue;
2260
2261 /* Clear the SRI_RECONF_SENT flag if too much time elapsed without
2262 * the slave moving forward to the next state. */
2263 if ((slave->flags & SRI_RECONF_SENT) &&
2264 (mstime() - slave->slave_reconf_sent_time) >
2265 SENTINEL_SLAVE_RECONF_RETRY_PERIOD)
2266 {
2267 sentinelEvent(REDIS_NOTICE,"-slave-reconf-sent-timeout",slave,"%@");
2268 slave->flags &= ~SRI_RECONF_SENT;
2269 }
2270
2271 /* Nothing to do for instances that are disconnected or already
2272 * in RECONF_SENT state. */
2273 if (slave->flags & (SRI_DISCONNECTED|SRI_RECONF_SENT|SRI_RECONF_INPROG))
2274 continue;
2275
2276 /* Send SLAVEOF <new master>. */
2277 ll2string(master_port,sizeof(master_port),
2278 master->promoted_slave->addr->port);
2279 retval = redisAsyncCommand(slave->cc,
2280 sentinelDiscardReplyCallback, NULL, "SLAVEOF %s %s",
2281 master->promoted_slave->addr->ip,
2282 master_port);
2283 if (retval == REDIS_OK) {
2284 slave->flags |= SRI_RECONF_SENT;
2285 slave->pending_commands++;
2286 slave->slave_reconf_sent_time = mstime();
2287 sentinelEvent(REDIS_NOTICE,"+slave-reconf-sent",slave,"%@");
2288 in_progress++;
2289 }
2290 }
2291 dictReleaseIterator(di);
2292 sentinelFailoverDetectEnd(master);
2293 }
2294
2295 /* This function is called when the slave is in
2296 * SENTINEL_FAILOVER_STATE_UPDATE_CONFIG state. In this state we need
2297 * to remove it from the master table and add the promoted slave instead.
2298 *
2299 * If there are no promoted slaves as this instance is unique, we remove
2300 * and re-add it with the same address to trigger a complete state
2301 * refresh. */
2302 void sentinelFailoverSwitchToPromotedSlave(sentinelRedisInstance *master) {
2303 sentinelRedisInstance *ref = master->promoted_slave ?
2304 master->promoted_slave : master;
2305
2306 sentinelEvent(REDIS_WARNING,"+switch-master",master,"%s %s %d %s %d",
2307 master->name, master->addr->ip, master->addr->port,
2308 ref->addr->ip, ref->addr->port);
2309
2310 sentinelResetMasterAndChangeAddress(master,ref->addr->ip,ref->addr->port);
2311 }
2312
2313 void sentinelFailoverStateMachine(sentinelRedisInstance *ri) {
2314 redisAssert(ri->flags & SRI_MASTER);
2315
2316 if (!(ri->flags & SRI_FAILOVER_IN_PROGRESS)) return;
2317
2318 switch(ri->failover_state) {
2319 case SENTINEL_FAILOVER_STATE_WAIT_START:
2320 sentinelFailoverWaitStart(ri);
2321 break;
2322 case SENTINEL_FAILOVER_STATE_SELECT_SLAVE:
2323 sentinelFailoverSelectSlave(ri);
2324 break;
2325 case SENTINEL_FAILOVER_STATE_SEND_SLAVEOF_NOONE:
2326 sentinelFailoverSendSlaveOfNoOne(ri);
2327 break;
2328 case SENTINEL_FAILOVER_STATE_WAIT_PROMOTION:
2329 sentinelFailoverWaitPromotion(ri);
2330 break;
2331 case SENTINEL_FAILOVER_STATE_RECONF_SLAVES:
2332 sentinelFailoverReconfNextSlave(ri);
2333 break;
2334 case SENTINEL_FAILOVER_STATE_DETECT_END:
2335 sentinelFailoverDetectEnd(ri);
2336 break;
2337 }
2338 }
2339
2340 /* The following is called only for master instances and will abort the
2341 * failover process if:
2342 *
2343 * 1) The failover is in progress.
2344 * 2) We already promoted a slave.
2345 * 3) The promoted slave is in extended SDOWN condition.
2346 */
2347 void sentinelAbortFailoverIfNeeded(sentinelRedisInstance *ri) {
2348 dictIterator *di;
2349 dictEntry *de;
2350
2351 /* Failover is in progress? Do we have a promoted slave? */
2352 if (!(ri->flags & SRI_FAILOVER_IN_PROGRESS) || !ri->promoted_slave) return;
2353
2354 /* Is the promoted slave into an extended SDOWN state? */
2355 if (!(ri->promoted_slave->flags & SRI_S_DOWN) ||
2356 (mstime() - ri->promoted_slave->s_down_since_time) <
2357 (ri->down_after_period * SENTINEL_EXTENDED_SDOWN_MULTIPLIER)) return;
2358
2359 sentinelEvent(REDIS_WARNING,"-failover-abort-x-sdown",ri->promoted_slave,"%@");
2360
2361 /* Clear failover related flags from slaves.
2362 * Also if we are the leader make sure to send SLAVEOF commands to all the
2363 * already reconfigured slaves in order to turn them back into slaves of
2364 * the original master. */
2365
2366 di = dictGetIterator(ri->slaves);
2367 while((de = dictNext(di)) != NULL) {
2368 sentinelRedisInstance *slave = dictGetVal(de);
2369 if (ri->flags & SRI_I_AM_THE_LEADER) {
2370 char master_port[32];
2371 int retval;
2372
2373 ll2string(master_port,sizeof(master_port),ri->addr->port);
2374 retval = redisAsyncCommand(slave->cc,
2375 sentinelDiscardReplyCallback, NULL, "SLAVEOF %s %s",
2376 ri->addr->ip,
2377 master_port);
2378 if (retval == REDIS_OK)
2379 sentinelEvent(REDIS_NOTICE,"-slave-reconf-undo",slave,"%@");
2380 }
2381 slave->flags &= ~(SRI_RECONF_SENT|SRI_RECONF_INPROG|SRI_RECONF_DONE);
2382 }
2383 dictReleaseIterator(di);
2384
2385 ri->flags &= ~(SRI_FAILOVER_IN_PROGRESS|SRI_I_AM_THE_LEADER);
2386 ri->failover_state = SENTINEL_FAILOVER_STATE_NONE;
2387 ri->failover_state_change_time = mstime();
2388 ri->promoted_slave->flags &= ~SRI_PROMOTED;
2389 ri->promoted_slave = NULL;
2390 }
2391
2392 /* ======================== SENTINEL timer handler ==========================
2393 * This is the "main" our Sentinel, being sentinel completely non blocking
2394 * in design. The function is called every second.
2395 * -------------------------------------------------------------------------- */
2396
2397 /* Perform scheduled operations for the specified Redis instance. */
2398 void sentinelHandleRedisInstance(sentinelRedisInstance *ri) {
2399 /* ========== MONITORING HALF ============ */
2400 /* Every kind of instance */
2401 sentinelReconnectInstance(ri);
2402 sentinelPingInstance(ri);
2403
2404 /* Masters and slaves */
2405 if (ri->flags & (SRI_MASTER|SRI_SLAVE)) {
2406 /* Nothing so far. */
2407 }
2408
2409 /* Only masters */
2410 if (ri->flags & SRI_MASTER) {
2411 sentinelAskMasterStateToOtherSentinels(ri);
2412 }
2413
2414 /* ============== ACTING HALF ============= */
2415 /* We don't proceed with the acting half if we are in TILT mode.
2416 * TILT happens when we find something odd with the time, like a
2417 * sudden change in the clock. */
2418 if (sentinel.tilt) {
2419 if (mstime()-sentinel.tilt_start_time < SENTINEL_TILT_PERIOD) return;
2420 sentinel.tilt = 0;
2421 sentinelEvent(REDIS_WARNING,"-tilt",NULL,"#tilt mode exited");
2422 }
2423
2424 /* Every kind of instance */
2425 sentinelCheckSubjectivelyDown(ri);
2426
2427 /* Masters and slaves */
2428 if (ri->flags & (SRI_MASTER|SRI_SLAVE)) {
2429 /* Nothing so far. */
2430 }
2431
2432 /* Only masters */
2433 if (ri->flags & SRI_MASTER) {
2434 sentinelCheckObjectivelyDown(ri);
2435 sentinelStartFailover(ri);
2436 sentinelFailoverStateMachine(ri);
2437 sentinelAbortFailoverIfNeeded(ri);
2438 }
2439 }
2440
2441 /* Perform scheduled operations for all the instances in the dictionary.
2442 * Recursively call the function against dictionaries of slaves. */
2443 void sentinelHandleDictOfRedisInstances(dict *instances) {
2444 dictIterator *di;
2445 dictEntry *de;
2446 sentinelRedisInstance *switch_to_promoted = NULL;
2447
2448 /* There are a number of things we need to perform against every master. */
2449 di = dictGetIterator(instances);
2450 while((de = dictNext(di)) != NULL) {
2451 sentinelRedisInstance *ri = dictGetVal(de);
2452
2453 sentinelHandleRedisInstance(ri);
2454 if (ri->flags & SRI_MASTER) {
2455 sentinelHandleDictOfRedisInstances(ri->slaves);
2456 sentinelHandleDictOfRedisInstances(ri->sentinels);
2457 if (ri->failover_state == SENTINEL_FAILOVER_STATE_UPDATE_CONFIG) {
2458 switch_to_promoted = ri;
2459 }
2460 }
2461 }
2462 if (switch_to_promoted)
2463 sentinelFailoverSwitchToPromotedSlave(switch_to_promoted);
2464 dictReleaseIterator(di);
2465 }
2466
2467 /* This function checks if we need to enter the TITL mode.
2468 *
2469 * The TILT mode is entered if we detect that between two invocations of the
2470 * timer interrupt, a negative amount of time, or too much time has passed.
2471 * Note that we expect that more or less just 100 milliseconds will pass
2472 * if everything is fine. However we'll see a negative number or a
2473 * difference bigger than SENTINEL_TILT_TRIGGER milliseconds if one of the
2474 * following conditions happen:
2475 *
2476 * 1) The Sentiel process for some time is blocked, for every kind of
2477 * random reason: the load is huge, the computer was freezed for some time
2478 * in I/O or alike, the process was stopped by a signal. Everything.
2479 * 2) The system clock was altered significantly.
2480 *
2481 * Under both this conditions we'll see everything as timed out and failing
2482 * without good reasons. Instead we enter the TILT mode and wait
2483 * for SENTIENL_TILT_PERIOD to elapse before starting to act again.
2484 *
2485 * During TILT time we still collect information, we just do not act. */
2486 void sentinelCheckTiltCondition(void) {
2487 mstime_t now = mstime();
2488 mstime_t delta = now - sentinel.previous_time;
2489
2490 if (delta < 0 || delta > SENTINEL_TILT_TRIGGER) {
2491 sentinel.tilt = 1;
2492 sentinel.tilt_start_time = mstime();
2493 sentinelEvent(REDIS_WARNING,"+tilt",NULL,"#tilt mode entered");
2494 }
2495 sentinel.previous_time = mstime();
2496 }
2497
2498 void sentinelTimer(void) {
2499 sentinelCheckTiltCondition();
2500 sentinelHandleDictOfRedisInstances(sentinel.masters);
2501 }
2502