network: don't unset the firewalld zone if it's going to be immediately re-set

Any time the firewalld zone for an interface is set, by definition
that removes it from any previous zone that it was in, so there is
really no point in unsetting the zone if it's just going to be
immediately set again.

This is useful because when firewalld reloads its rules, 3 things happen:

1) firewalld flushes *all* firewall rules (including those added by libvirt)

2) firewalld unsets the zones for all interfaces (including those set
   by libvirt)

3) firewalld re-adds its own rules, and sets the zone for all the
   interfaces it manages

4) firewalld sends a dbus message that libvirt is watching for, and
   when libvirt receives that message, it reloads all of the
   libvirt-generated rules, and also re-sets the firewalld zone for
   the bridge interfaces managed by libvirt.

libvirt accomplishes step 4 by a) calling
networkRemoveFirewallRules(), and then b) calling
networkAddFirewallRules(). But (because it is useful in other
contexts) networkRemoveFirewallRules() will attempt to *unset* the
zone for each bridge interface, and when firewalld receives this
request, it sees that the bridge interface *has no zone* (because it
was unset by firewalld in step (2) above), and thus logs an error
message.

There is no way for libvirt to suppress an error message that is
logged by firewalld when a request to firewalld fails. But what
libvirt *can* do is realize that in these cases, the firewalld zone is
about to be set again anyway, and so we don't need to unset the zone.

This patch handles that by adding a bool unsetZone to the arguments of
networkRemoveFirewallRules(); most calls to networkRemoveFirewallRules()
have unsetZone=true, but in two cases where the zone is about to be
reset, networkRemoveFirewallRules() is called with unsetZone=false,
which prevents the call to virFirewallDInterfaceUnsetZone() and thus
avoids the unnecessary (and confusing to users!) error message that
would have been logged by firewalld.

Signed-off-by: Laine Stump <laine@redat.com>
Reviewed-by: Ján Tomko <jtomko@redhat.com>
This commit is contained in:
Laine Stump 2024-10-21 21:31:37 -04:00
parent e8228a9e79
commit d5af1e90bb
4 changed files with 15 additions and 10 deletions

View File

@ -1752,7 +1752,7 @@ networkReloadFirewallRulesHelper(virNetworkObj *obj,
* and this functionality is also handled by * and this functionality is also handled by
* networkAdd/RemoveFirewallRules() * networkAdd/RemoveFirewallRules()
*/ */
networkRemoveFirewallRules(obj); networkRemoveFirewallRules(obj, false);
ignore_value(networkAddFirewallRules(def, cfg->firewallBackend, &fwRemoval)); ignore_value(networkAddFirewallRules(def, cfg->firewallBackend, &fwRemoval));
virNetworkObjSetFwRemoval(obj, fwRemoval); virNetworkObjSetFwRemoval(obj, fwRemoval);
saveStatus = true; saveStatus = true;
@ -2129,7 +2129,7 @@ networkStartNetworkVirtual(virNetworkDriverState *driver,
ignore_value(virNetDevSetOnline(def->bridge, false)); ignore_value(virNetDevSetOnline(def->bridge, false));
if (firewalRulesAdded) if (firewalRulesAdded)
networkRemoveFirewallRules(obj); networkRemoveFirewallRules(obj, true);
virNetworkObjUnrefMacMap(obj); virNetworkObjUnrefMacMap(obj);
@ -2166,7 +2166,7 @@ networkShutdownNetworkVirtual(virNetworkObj *obj)
ignore_value(virNetDevSetOnline(def->bridge, false)); ignore_value(virNetDevSetOnline(def->bridge, false));
networkRemoveFirewallRules(obj); networkRemoveFirewallRules(obj, true);
ignore_value(virNetDevBridgeDelete(def->bridge)); ignore_value(virNetDevBridgeDelete(def->bridge));
@ -3332,7 +3332,7 @@ networkUpdate(virNetworkPtr net,
* old rules (and remember to load new ones after the * old rules (and remember to load new ones after the
* update). * update).
*/ */
networkRemoveFirewallRules(obj); networkRemoveFirewallRules(obj, false);
needFirewallRefresh = true; needFirewallRefresh = true;
break; break;
default: default:

View File

@ -447,7 +447,8 @@ networkAddFirewallRules(virNetworkDef *def,
void void
networkRemoveFirewallRules(virNetworkObj *obj) networkRemoveFirewallRules(virNetworkObj *obj,
bool unsetZone)
{ {
virNetworkDef *def = virNetworkObjGetDef(obj); virNetworkDef *def = virNetworkObjGetDef(obj);
virFirewall *fw; virFirewall *fw;
@ -484,9 +485,10 @@ networkRemoveFirewallRules(virNetworkObj *obj)
* same interface name wants *no* zone set. To avoid this, we must * same interface name wants *no* zone set. To avoid this, we must
* "unset" the zone if we set it when the network was started. * "unset" the zone if we set it when the network was started.
*/ */
if (virFirewallDIsRegistered() == 0 && if (unsetZone
(def->forward.type != VIR_NETWORK_FORWARD_OPEN || && virFirewallDIsRegistered() == 0
def->bridgeZone)) { && (def->forward.type != VIR_NETWORK_FORWARD_OPEN
|| def->bridgeZone)) {
VIR_DEBUG("unsetting zone for '%s' (current zone is '%s')", VIR_DEBUG("unsetting zone for '%s' (current zone is '%s')",
def->bridge, def->bridgeZone); def->bridge, def->bridgeZone);

View File

@ -56,6 +56,8 @@ int networkAddFirewallRules(virNetworkDef *def G_GNUC_UNUSED,
return 0; return 0;
} }
void networkRemoveFirewallRules(virNetworkObj *obj G_GNUC_UNUSED) void
networkRemoveFirewallRules(virNetworkObj *obj G_GNUC_UNUSED,
bool unsetZone G_GNUC_UNUSED)
{ {
} }

View File

@ -36,4 +36,5 @@ int networkAddFirewallRules(virNetworkDef *def,
virFirewallBackend firewallBackend, virFirewallBackend firewallBackend,
virFirewall **fwRemoval); virFirewall **fwRemoval);
void networkRemoveFirewallRules(virNetworkObj *obj); void networkRemoveFirewallRules(virNetworkObj *obj,
bool unsetZone);