mirror of
https://github.com/samba-team/samba.git
synced 2025-01-08 21:18:16 +03:00
ctdbd: Stop takeovers and releases from colliding in mid-air
There's a race here where release and takeover events for an IP can run at the same time. For example, a "ctdb deleteip" and a takeover initiated by the recovery daemon. The timeline is as follows: 1. The release code registers a callback to update the VNN. The callback is executed *after* the eventscripts run the releaseip event. 2. The release code calls the eventscripts for the releaseip event, removing IP from its interface. The takeover code "updates" the VNN saying that IP is on some iface.... even if/though the address is already there. 3. The release callback runs, removing the iface associated with IP in the VNN. The takeover code calls the eventscripts for the takeip event, adding IP to an interface. As a result, CTDB doesn't think it should be hosting IP but IP is on an interface. The recovery daemon fixes this later... but it shouldn't happen. This patch can cause some additional noise in the logs: Release of IP 10.0.2.133/24 on interface eth2 node:2 recoverd:We are still serving a public address '10.0.2.133' that we should not be serving. Removing it. Release of IP 10.0.2.133/24 rejected update for this IP already in flight recoverd:client/ctdb_client.c:2455 ctdb_control for release_ip failed recoverd:Failed to release local ip address In this case the node has started releasing an IP when the recovery daemon notices the addresses is still hosted and initiates another release. This noise is harmless but annoying. Signed-off-by: Martin Schwenke <martin@meltin.net> (This used to be ctdb commit bfe16cf69bf2eee93c0d831f76d88bba0c2b96c2)
This commit is contained in:
parent
79ea15bf96
commit
4b4e4d8870
@ -213,6 +213,10 @@ struct ctdb_vnn {
|
|||||||
TALLOC_CTX *takeover_ctx;
|
TALLOC_CTX *takeover_ctx;
|
||||||
|
|
||||||
struct ctdb_kill_tcp *killtcp;
|
struct ctdb_kill_tcp *killtcp;
|
||||||
|
|
||||||
|
/* Set to true any time an update to this VNN is in flight.
|
||||||
|
This helps to avoid races. */
|
||||||
|
bool update_in_flight;
|
||||||
};
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -373,6 +373,12 @@ static void ctdb_do_takeip_callback(struct ctdb_context *ctdb, int status,
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int ctdb_takeip_destructor(struct ctdb_do_takeip_state *state)
|
||||||
|
{
|
||||||
|
state->vnn->update_in_flight = false;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
take over an ip address
|
take over an ip address
|
||||||
*/
|
*/
|
||||||
@ -383,6 +389,14 @@ static int32_t ctdb_do_takeip(struct ctdb_context *ctdb,
|
|||||||
int ret;
|
int ret;
|
||||||
struct ctdb_do_takeip_state *state;
|
struct ctdb_do_takeip_state *state;
|
||||||
|
|
||||||
|
if (vnn->update_in_flight) {
|
||||||
|
DEBUG(DEBUG_NOTICE,("Takeover of IP %s/%u rejected "
|
||||||
|
"update for this IP already in flight\n",
|
||||||
|
ctdb_addr_to_str(&vnn->public_address),
|
||||||
|
vnn->public_netmask_bits));
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
ret = ctdb_vnn_assign_iface(ctdb, vnn);
|
ret = ctdb_vnn_assign_iface(ctdb, vnn);
|
||||||
if (ret != 0) {
|
if (ret != 0) {
|
||||||
DEBUG(DEBUG_ERR,("Takeover of IP %s/%u failed to "
|
DEBUG(DEBUG_ERR,("Takeover of IP %s/%u failed to "
|
||||||
@ -398,6 +412,9 @@ static int32_t ctdb_do_takeip(struct ctdb_context *ctdb,
|
|||||||
state->c = talloc_steal(ctdb, c);
|
state->c = talloc_steal(ctdb, c);
|
||||||
state->vnn = vnn;
|
state->vnn = vnn;
|
||||||
|
|
||||||
|
vnn->update_in_flight = true;
|
||||||
|
talloc_set_destructor(state, ctdb_takeip_destructor);
|
||||||
|
|
||||||
DEBUG(DEBUG_NOTICE,("Takeover of IP %s/%u on interface %s\n",
|
DEBUG(DEBUG_NOTICE,("Takeover of IP %s/%u on interface %s\n",
|
||||||
ctdb_addr_to_str(&vnn->public_address),
|
ctdb_addr_to_str(&vnn->public_address),
|
||||||
vnn->public_netmask_bits,
|
vnn->public_netmask_bits,
|
||||||
@ -480,6 +497,12 @@ static void ctdb_do_updateip_callback(struct ctdb_context *ctdb, int status,
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int ctdb_updateip_destructor(struct ctdb_do_updateip_state *state)
|
||||||
|
{
|
||||||
|
state->vnn->update_in_flight = false;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
update (move) an ip address
|
update (move) an ip address
|
||||||
*/
|
*/
|
||||||
@ -492,6 +515,14 @@ static int32_t ctdb_do_updateip(struct ctdb_context *ctdb,
|
|||||||
struct ctdb_iface *old = vnn->iface;
|
struct ctdb_iface *old = vnn->iface;
|
||||||
const char *new_name;
|
const char *new_name;
|
||||||
|
|
||||||
|
if (vnn->update_in_flight) {
|
||||||
|
DEBUG(DEBUG_NOTICE,("Update of IP %s/%u rejected "
|
||||||
|
"update for this IP already in flight\n",
|
||||||
|
ctdb_addr_to_str(&vnn->public_address),
|
||||||
|
vnn->public_netmask_bits));
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
ctdb_vnn_unassign_iface(ctdb, vnn);
|
ctdb_vnn_unassign_iface(ctdb, vnn);
|
||||||
ret = ctdb_vnn_assign_iface(ctdb, vnn);
|
ret = ctdb_vnn_assign_iface(ctdb, vnn);
|
||||||
if (ret != 0) {
|
if (ret != 0) {
|
||||||
@ -520,6 +551,9 @@ static int32_t ctdb_do_updateip(struct ctdb_context *ctdb,
|
|||||||
state->old = old;
|
state->old = old;
|
||||||
state->vnn = vnn;
|
state->vnn = vnn;
|
||||||
|
|
||||||
|
vnn->update_in_flight = true;
|
||||||
|
talloc_set_destructor(state, ctdb_updateip_destructor);
|
||||||
|
|
||||||
DEBUG(DEBUG_NOTICE,("Update of IP %s/%u from "
|
DEBUG(DEBUG_NOTICE,("Update of IP %s/%u from "
|
||||||
"interface %s to %s\n",
|
"interface %s to %s\n",
|
||||||
ctdb_addr_to_str(&vnn->public_address),
|
ctdb_addr_to_str(&vnn->public_address),
|
||||||
@ -782,6 +816,12 @@ static void release_ip_callback(struct ctdb_context *ctdb, int status,
|
|||||||
talloc_free(state);
|
talloc_free(state);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int ctdb_releaseip_destructor(struct takeover_callback_state *state)
|
||||||
|
{
|
||||||
|
state->vnn->update_in_flight = false;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
release an ip address
|
release an ip address
|
||||||
*/
|
*/
|
||||||
@ -809,8 +849,12 @@ int32_t ctdb_control_release_ip(struct ctdb_context *ctdb,
|
|||||||
talloc_free(vnn->takeover_ctx);
|
talloc_free(vnn->takeover_ctx);
|
||||||
vnn->takeover_ctx = NULL;
|
vnn->takeover_ctx = NULL;
|
||||||
|
|
||||||
|
/* Some ctdb tool commands (e.g. moveip, rebalanceip) send
|
||||||
|
* lazy multicast to drop an IP from any node that isn't the
|
||||||
|
* intended new node. The following causes makes ctdbd ignore
|
||||||
|
* a release for any address it doesn't host.
|
||||||
|
*/
|
||||||
if (ctdb->do_checkpublicip) {
|
if (ctdb->do_checkpublicip) {
|
||||||
|
|
||||||
if (!ctdb_sys_have_ip(&pip->addr)) {
|
if (!ctdb_sys_have_ip(&pip->addr)) {
|
||||||
DEBUG(DEBUG_DEBUG,("Redundant release of IP %s/%u on interface %s (ip not held)\n",
|
DEBUG(DEBUG_DEBUG,("Redundant release of IP %s/%u on interface %s (ip not held)\n",
|
||||||
ctdb_addr_to_str(&pip->addr),
|
ctdb_addr_to_str(&pip->addr),
|
||||||
@ -819,12 +863,6 @@ int32_t ctdb_control_release_ip(struct ctdb_context *ctdb,
|
|||||||
ctdb_vnn_unassign_iface(ctdb, vnn);
|
ctdb_vnn_unassign_iface(ctdb, vnn);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
iface = ctdb_sys_find_ifname(&pip->addr);
|
|
||||||
if (iface == NULL) {
|
|
||||||
DEBUG(DEBUG_ERR, ("Could not find which interface the ip address is hosted on. can not release it\n"));
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
if (vnn->iface == NULL) {
|
if (vnn->iface == NULL) {
|
||||||
DEBUG(DEBUG_DEBUG,("Redundant release of IP %s/%u (ip not held)\n",
|
DEBUG(DEBUG_DEBUG,("Redundant release of IP %s/%u (ip not held)\n",
|
||||||
@ -832,6 +870,28 @@ int32_t ctdb_control_release_ip(struct ctdb_context *ctdb,
|
|||||||
vnn->public_netmask_bits));
|
vnn->public_netmask_bits));
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* There is a potential race between take_ip and us because we
|
||||||
|
* update the VNN via a callback that run when the
|
||||||
|
* eventscripts have been run. Avoid the race by allowing one
|
||||||
|
* update to be in flight at a time.
|
||||||
|
*/
|
||||||
|
if (vnn->update_in_flight) {
|
||||||
|
DEBUG(DEBUG_NOTICE,("Release of IP %s/%u rejected "
|
||||||
|
"update for this IP already in flight\n",
|
||||||
|
ctdb_addr_to_str(&vnn->public_address),
|
||||||
|
vnn->public_netmask_bits));
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ctdb->do_checkpublicip) {
|
||||||
|
iface = ctdb_sys_find_ifname(&pip->addr);
|
||||||
|
if (iface == NULL) {
|
||||||
|
DEBUG(DEBUG_ERR, ("Could not find which interface the ip address is hosted on. can not release it\n"));
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
iface = strdup(ctdb_vnn_iface_string(vnn));
|
iface = strdup(ctdb_vnn_iface_string(vnn));
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -850,6 +910,9 @@ int32_t ctdb_control_release_ip(struct ctdb_context *ctdb,
|
|||||||
*state->addr = pip->addr;
|
*state->addr = pip->addr;
|
||||||
state->vnn = vnn;
|
state->vnn = vnn;
|
||||||
|
|
||||||
|
vnn->update_in_flight = true;
|
||||||
|
talloc_set_destructor(state, ctdb_releaseip_destructor);
|
||||||
|
|
||||||
ret = ctdb_event_script_callback(ctdb,
|
ret = ctdb_event_script_callback(ctdb,
|
||||||
state, release_ip_callback, state,
|
state, release_ip_callback, state,
|
||||||
false,
|
false,
|
||||||
|
Loading…
Reference in New Issue
Block a user