mirror of
https://codeberg.org/redict/redict.git
synced 2025-01-23 00:28:26 -05:00
Merge pull request #5201 from soloestoy/abort-if-lost-exec-aof-load
AOF: discard if we lost EXEC when loading aof
This commit is contained in:
commit
1a476b69ad
15
src/aof.c
15
src/aof.c
@ -677,6 +677,7 @@ int loadAppendOnlyFile(char *filename) {
|
|||||||
int old_aof_state = server.aof_state;
|
int old_aof_state = server.aof_state;
|
||||||
long loops = 0;
|
long loops = 0;
|
||||||
off_t valid_up_to = 0; /* Offset of latest well-formed command loaded. */
|
off_t valid_up_to = 0; /* Offset of latest well-formed command loaded. */
|
||||||
|
off_t valid_before_multi = 0; /* Offset before MULTI command loaded. */
|
||||||
|
|
||||||
if (fp == NULL) {
|
if (fp == NULL) {
|
||||||
serverLog(LL_WARNING,"Fatal error: can't open the append log file for reading: %s",strerror(errno));
|
serverLog(LL_WARNING,"Fatal error: can't open the append log file for reading: %s",strerror(errno));
|
||||||
@ -781,9 +782,15 @@ int loadAppendOnlyFile(char *filename) {
|
|||||||
exit(1);
|
exit(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (cmd == server.multiCommand) valid_before_multi = valid_up_to;
|
||||||
|
|
||||||
/* Run the command in the context of a fake client */
|
/* Run the command in the context of a fake client */
|
||||||
fakeClient->cmd = cmd;
|
fakeClient->cmd = cmd;
|
||||||
cmd->proc(fakeClient);
|
if (fakeClient->flags & CLIENT_MULTI && fakeClient->cmd->proc != execCommand) {
|
||||||
|
queueMultiCommand(fakeClient);
|
||||||
|
} else {
|
||||||
|
cmd->proc(fakeClient);
|
||||||
|
}
|
||||||
|
|
||||||
/* The fake client should not have a reply */
|
/* The fake client should not have a reply */
|
||||||
serverAssert(fakeClient->bufpos == 0 && listLength(fakeClient->reply) == 0);
|
serverAssert(fakeClient->bufpos == 0 && listLength(fakeClient->reply) == 0);
|
||||||
@ -801,7 +808,11 @@ int loadAppendOnlyFile(char *filename) {
|
|||||||
* If the client is in the middle of a MULTI/EXEC, handle it as it was
|
* If the client is in the middle of a MULTI/EXEC, handle it as it was
|
||||||
* a short read, even if technically the protocol is correct: we want
|
* a short read, even if technically the protocol is correct: we want
|
||||||
* to remove the unprocessed tail and continue. */
|
* to remove the unprocessed tail and continue. */
|
||||||
if (fakeClient->flags & CLIENT_MULTI) goto uxeof;
|
if (fakeClient->flags & CLIENT_MULTI) {
|
||||||
|
serverLog(LL_WARNING,"!!! Warning: we lost EXEC in the middle of transaction, discard !!!");
|
||||||
|
valid_up_to = valid_before_multi;
|
||||||
|
goto uxeof;
|
||||||
|
}
|
||||||
|
|
||||||
loaded_ok: /* DB loaded, cleanup and return C_OK to the caller. */
|
loaded_ok: /* DB loaded, cleanup and return C_OK to the caller. */
|
||||||
fclose(fp);
|
fclose(fp);
|
||||||
|
@ -158,7 +158,7 @@ void execCommand(client *c) {
|
|||||||
must_propagate = 1;
|
must_propagate = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
call(c,CMD_CALL_FULL);
|
call(c,server.loading ? CMD_CALL_NONE : CMD_CALL_FULL);
|
||||||
|
|
||||||
/* Commands may alter argc/argv, restore mstate. */
|
/* Commands may alter argc/argv, restore mstate. */
|
||||||
c->mstate.commands[j].argc = c->argc;
|
c->mstate.commands[j].argc = c->argc;
|
||||||
|
Loading…
Reference in New Issue
Block a user