diff --git a/changelogs/unreleased/gh-5819-election-triggers.md b/changelogs/unreleased/gh-5819-election-triggers.md new file mode 100644 index 0000000000000000000000000000000000000000..edb7be8ac2ba6212ae4067a2554a3587e12a0df5 --- /dev/null +++ b/changelogs/unreleased/gh-5819-election-triggers.md @@ -0,0 +1,5 @@ +## feature/replication + +* Introduce on_election triggers. The triggers may be registered via +`box.ctl.on_election()` interface and are run asynchronously each time +`box.info.election` changes (gh-5819). diff --git a/src/box/lua/ctl.c b/src/box/lua/ctl.c index 368b9ab6098d39174248f3a8d8863ef699d014bd..4a9212f333f31a52d450c17db1b4cfdd31a8f51d 100644 --- a/src/box/lua/ctl.c +++ b/src/box/lua/ctl.c @@ -43,6 +43,7 @@ #include "box/schema.h" #include "box/engine.h" #include "box/memtx_engine.h" +#include "box/raft.h" static int lbox_ctl_wait_ro(struct lua_State *L) @@ -81,6 +82,12 @@ lbox_ctl_on_schema_init(struct lua_State *L) return lbox_trigger_reset(L, 2, &on_schema_init, NULL, NULL); } +static int +lbox_ctl_on_election(struct lua_State *L) +{ + return lbox_trigger_reset(L, 2, &box_raft_on_broadcast, NULL, NULL); +} + static int lbox_ctl_promote(struct lua_State *L) { @@ -124,6 +131,7 @@ static const struct luaL_Reg lbox_ctl_lib[] = { {"wait_rw", lbox_ctl_wait_rw}, {"on_shutdown", lbox_ctl_on_shutdown}, {"on_schema_init", lbox_ctl_on_schema_init}, + {"on_election", lbox_ctl_on_election}, {"promote", lbox_ctl_promote}, /* An old alias. */ {"clear_synchro_queue", lbox_ctl_promote}, diff --git a/src/box/raft.c b/src/box/raft.c index eb62e96300f840db92d522d1ec0d4fe328e0b150..563ed89c4af6ddac250dc9ec96346c0edef1cfd2 100644 --- a/src/box/raft.c +++ b/src/box/raft.c @@ -52,6 +52,9 @@ enum election_mode box_election_mode = ELECTION_MODE_INVALID; */ static struct trigger box_raft_on_update; +struct rlist box_raft_on_broadcast = + RLIST_HEAD_INITIALIZER(box_raft_on_broadcast); + /** * Worker fiber does all the asynchronous work, which may need yields and can be * long. These are WAL writes, network broadcasts. That allows not to block the @@ -276,6 +279,7 @@ box_raft_broadcast(struct raft *raft, const struct raft_msg *msg) box_raft_msg_to_request(msg, &req); replicaset_foreach(replica) relay_push_raft(replica->relay, &req); + trigger_run(&box_raft_on_broadcast, NULL); } static void diff --git a/src/box/raft.h b/src/box/raft.h index d55e90d2fa2007f55055c4da41a2c7fe89664019..b4a97444af9173d9cd18cd6448d2193d62b651e1 100644 --- a/src/box/raft.h +++ b/src/box/raft.h @@ -30,11 +30,18 @@ * SUCH DAMAGE. */ #include "raft/raft.h" +#include "small/rlist.h" #if defined(__cplusplus) extern "C" { #endif +/** + * A public trigger fired on Raft state change, i.e. on a broadcast. + * It's allowed to yield inside it, and it's run asynchronously. + */ +extern struct rlist box_raft_on_broadcast; + enum election_mode { ELECTION_MODE_INVALID = -1, ELECTION_MODE_OFF = 0, diff --git a/test/replication/election_basic.result b/test/replication/election_basic.result index b64028c6029bf97f92c96bb4e1a99a621c946586..fc7121d7f8ac6f116122ead6f8aefd456f4e27d1 100644 --- a/test/replication/election_basic.result +++ b/test/replication/election_basic.result @@ -275,3 +275,147 @@ test_run:cmd(string.format('start server %s', leader_name)) test_run:drop_cluster(SERVERS) | --- | ... + +-- gh-5819: on_election triggers, that are filed on every visible state change. +box.schema.user.grant('guest', 'replication') + | --- + | ... +test_run:cmd('create server replica with rpl_master=default,\ + script="replication/replica.lua"') + | --- + | - true + | ... +test_run:cmd('start server replica') + | --- + | - true + | ... + +repl = test_run:eval('replica', 'return box.cfg.listen')[1] + | --- + | ... +box.cfg{replication = repl} + | --- + | ... +test_run:switch('replica') + | --- + | - true + | ... +box.cfg{election_mode='voter'} + | --- + | ... +test_run:switch('default') + | --- + | - true + | ... + +election_tbl = {} + | --- + | ... + +function trig()\ + election_tbl[#election_tbl+1] = box.info.election\ +end + | --- + | ... + +_ = box.ctl.on_election(trig) + | --- + | ... + +box.cfg{replication_synchro_quorum=2} + | --- + | ... +box.cfg{election_mode='candidate'} + | --- + | ... + +test_run:wait_cond(function() return #election_tbl == 3 end) + | --- + | - true + | ... +assert(election_tbl[1].state == 'follower') + | --- + | - true + | ... +assert(election_tbl[2].state == 'candidate') + | --- + | - true + | ... +assert(election_tbl[2].vote == 1) + | --- + | - true + | ... +assert(election_tbl[3].state == 'leader') + | --- + | - true + | ... + +box.cfg{election_mode='voter'} + | --- + | ... +test_run:wait_cond(function() return #election_tbl == 4 end) + | --- + | - true + | ... +assert(election_tbl[4].state == 'follower') + | --- + | - true + | ... + +box.cfg{election_mode='off'} + | --- + | ... +test_run:wait_cond(function() return #election_tbl == 5 end) + | --- + | - true + | ... + +box.cfg{election_mode='manual'} + | --- + | ... +test_run:wait_cond(function() return #election_tbl == 6 end) + | --- + | - true + | ... +assert(election_tbl[6].state == 'follower') + | --- + | - true + | ... + +box.ctl.promote() + | --- + | ... + +test_run:wait_cond(function() return #election_tbl == 9 end) + | --- + | - true + | ... +assert(election_tbl[7].state == 'follower') + | --- + | - true + | ... +assert(election_tbl[7].term == election_tbl[6].term + 1) + | --- + | - true + | ... +assert(election_tbl[8].state == 'candidate') + | --- + | - true + | ... +assert(election_tbl[9].state == 'leader') + | --- + | - true + | ... + +test_run:cmd('stop server replica') + | --- + | - true + | ... +test_run:cmd('delete server replica') + | --- + | - true + | ... + +box.schema.user.revoke('guest', 'replication') + | --- + | ... diff --git a/test/replication/election_basic.test.lua b/test/replication/election_basic.test.lua index 77fdf6340e2efb895a741f6f30ded825a7e2da6a..f1330d232931a26fe6a0bf90e160ff212cf4a373 100644 --- a/test/replication/election_basic.test.lua +++ b/test/replication/election_basic.test.lua @@ -116,3 +116,56 @@ assert(r1_leader == r2_leader) test_run:cmd(string.format('start server %s', leader_name)) test_run:drop_cluster(SERVERS) + +-- gh-5819: on_election triggers, that are filed on every visible state change. +box.schema.user.grant('guest', 'replication') +test_run:cmd('create server replica with rpl_master=default,\ + script="replication/replica.lua"') +test_run:cmd('start server replica') + +repl = test_run:eval('replica', 'return box.cfg.listen')[1] +box.cfg{replication = repl} +test_run:switch('replica') +box.cfg{election_mode='voter'} +test_run:switch('default') + +election_tbl = {} + +function trig()\ + election_tbl[#election_tbl+1] = box.info.election\ +end + +_ = box.ctl.on_election(trig) + +box.cfg{replication_synchro_quorum=2} +box.cfg{election_mode='candidate'} + +test_run:wait_cond(function() return #election_tbl == 3 end) +assert(election_tbl[1].state == 'follower') +assert(election_tbl[2].state == 'candidate') +assert(election_tbl[2].vote == 1) +assert(election_tbl[3].state == 'leader') + +box.cfg{election_mode='voter'} +test_run:wait_cond(function() return #election_tbl == 4 end) +assert(election_tbl[4].state == 'follower') + +box.cfg{election_mode='off'} +test_run:wait_cond(function() return #election_tbl == 5 end) + +box.cfg{election_mode='manual'} +test_run:wait_cond(function() return #election_tbl == 6 end) +assert(election_tbl[6].state == 'follower') + +box.ctl.promote() + +test_run:wait_cond(function() return #election_tbl == 9 end) +assert(election_tbl[7].state == 'follower') +assert(election_tbl[7].term == election_tbl[6].term + 1) +assert(election_tbl[8].state == 'candidate') +assert(election_tbl[9].state == 'leader') + +test_run:cmd('stop server replica') +test_run:cmd('delete server replica') + +box.schema.user.revoke('guest', 'replication')