2019-01-09 19:11:32 +00:00
|
|
|
/*
|
|
|
|
* virfirewalld.c: support for firewalld (https://firewalld.org)
|
|
|
|
*
|
|
|
|
* Copyright (C) 2019 Red Hat, Inc.
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library. If not, see
|
|
|
|
* <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
|
|
|
#include <stdarg.h>
|
|
|
|
|
|
|
|
#include "virfirewall.h"
|
|
|
|
#include "virfirewalld.h"
|
|
|
|
#define LIBVIRT_VIRFIREWALLDPRIV_H_ALLOW
|
|
|
|
#include "virfirewalldpriv.h"
|
2019-04-01 14:28:05 +00:00
|
|
|
#include "viralloc.h"
|
2019-01-09 19:11:32 +00:00
|
|
|
#include "virerror.h"
|
|
|
|
#include "virutil.h"
|
|
|
|
#include "virlog.h"
|
|
|
|
#include "virdbus.h"
|
|
|
|
|
|
|
|
#define VIR_FROM_THIS VIR_FROM_FIREWALLD
|
|
|
|
|
|
|
|
VIR_LOG_INIT("util.firewalld");
|
|
|
|
|
|
|
|
/* used to convert virFirewallLayer enum values to strings
|
|
|
|
* understood by the firewalld.direct "passthrough" method
|
|
|
|
*/
|
|
|
|
VIR_ENUM_DECL(virFirewallLayerFirewallD);
|
|
|
|
VIR_ENUM_IMPL(virFirewallLayerFirewallD, VIR_FIREWALL_LAYER_LAST,
|
|
|
|
"eb",
|
|
|
|
"ipv4",
|
|
|
|
"ipv6",
|
|
|
|
);
|
|
|
|
|
|
|
|
|
2019-01-09 19:40:51 +00:00
|
|
|
VIR_ENUM_DECL(virFirewallDBackend);
|
|
|
|
VIR_ENUM_IMPL(virFirewallDBackend, VIR_FIREWALLD_BACKEND_LAST,
|
|
|
|
"",
|
|
|
|
"iptables",
|
|
|
|
"nftables",
|
|
|
|
);
|
|
|
|
|
|
|
|
|
2019-01-09 19:11:32 +00:00
|
|
|
/**
|
|
|
|
* virFirewallDIsRegistered:
|
|
|
|
*
|
|
|
|
* Returns 0 if service is registered, -1 on fatal error, or -2 if service is not registered
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
virFirewallDIsRegistered(void)
|
|
|
|
{
|
|
|
|
return virDBusIsServiceRegistered(VIR_FIREWALL_FIREWALLD_SERVICE);
|
|
|
|
}
|
|
|
|
|
2019-01-09 19:40:51 +00:00
|
|
|
/**
|
|
|
|
* virFirewallDGetVersion:
|
|
|
|
* @version: pointer to location to save version in the form of:
|
|
|
|
* 1000000 * major + 1000 * minor + micro
|
|
|
|
*
|
|
|
|
* queries the firewalld version property from dbus, and converts it
|
|
|
|
* from a string into a number.
|
|
|
|
*
|
|
|
|
* Returns 0 if version was successfully retrieved, or -1 on error
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
virFirewallDGetVersion(unsigned long *version)
|
|
|
|
{
|
|
|
|
int ret = -1;
|
|
|
|
DBusConnection *sysbus = virDBusGetSystemBus();
|
|
|
|
DBusMessage *reply = NULL;
|
|
|
|
VIR_AUTOFREE(char *) versionStr = NULL;
|
|
|
|
|
|
|
|
if (!sysbus)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (virDBusCallMethod(sysbus,
|
|
|
|
&reply,
|
|
|
|
NULL,
|
|
|
|
VIR_FIREWALL_FIREWALLD_SERVICE,
|
|
|
|
"/org/fedoraproject/FirewallD1",
|
|
|
|
"org.freedesktop.DBus.Properties",
|
|
|
|
"Get",
|
|
|
|
"ss",
|
|
|
|
"org.fedoraproject.FirewallD1",
|
|
|
|
"version") < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (virDBusMessageRead(reply, "v", "s", &versionStr) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (virParseVersionString(versionStr, version, false) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Failed to parse firewalld version '%s'"),
|
|
|
|
versionStr);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
VIR_DEBUG("FirewallD version: %s - %lu", versionStr, *version);
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
|
|
|
virDBusMessageUnref(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* virFirewallDGetBackend:
|
|
|
|
*
|
|
|
|
* Returns virVirewallDBackendType value representing which packet
|
|
|
|
* filtering backend is currently in use by firewalld, or -1 on error.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
virFirewallDGetBackend(void)
|
|
|
|
{
|
|
|
|
DBusConnection *sysbus = virDBusGetSystemBus();
|
|
|
|
DBusMessage *reply = NULL;
|
|
|
|
virError error;
|
|
|
|
VIR_AUTOFREE(char *) backendStr = NULL;
|
|
|
|
int backend = -1;
|
|
|
|
|
|
|
|
if (!sysbus)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
memset(&error, 0, sizeof(error));
|
|
|
|
|
|
|
|
if (virDBusCallMethod(sysbus,
|
|
|
|
&reply,
|
|
|
|
&error,
|
|
|
|
VIR_FIREWALL_FIREWALLD_SERVICE,
|
|
|
|
"/org/fedoraproject/FirewallD1/config",
|
|
|
|
"org.freedesktop.DBus.Properties",
|
|
|
|
"Get",
|
|
|
|
"ss",
|
|
|
|
"org.fedoraproject.FirewallD1.config",
|
|
|
|
"FirewallBackend") < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (error.level == VIR_ERR_ERROR) {
|
|
|
|
/* we don't want to log any error in the case that
|
|
|
|
* FirewallBackend isn't implemented in this firewalld, since
|
|
|
|
* that just means that it is an old version, and only has an
|
|
|
|
* iptables backend.
|
|
|
|
*/
|
|
|
|
VIR_DEBUG("Failed to get FirewallBackend setting, assuming 'iptables'");
|
|
|
|
backend = VIR_FIREWALLD_BACKEND_IPTABLES;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virDBusMessageRead(reply, "v", "s", &backendStr) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
VIR_DEBUG("FirewallD backend: %s", backendStr);
|
|
|
|
|
|
|
|
if ((backend = virFirewallDBackendTypeFromString(backendStr)) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Unrecognized firewalld backend type: %s"),
|
|
|
|
backendStr);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
virResetError(&error);
|
|
|
|
virDBusMessageUnref(reply);
|
|
|
|
return backend;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* virFirewallDGetZones:
|
|
|
|
* @zones: array of char *, each entry is a null-terminated zone name
|
|
|
|
* @nzones: number of entries in @zones
|
|
|
|
*
|
|
|
|
* Get the number of currently active firewalld zones, and their names
|
|
|
|
* in an array of null-terminated strings. The memory pointed to by
|
|
|
|
* @zones will belong to the caller, and must be freed.
|
|
|
|
*
|
|
|
|
* Returns 0 on success, -1 (and failure logged) on error
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
virFirewallDGetZones(char ***zones, size_t *nzones)
|
|
|
|
{
|
|
|
|
DBusConnection *sysbus = virDBusGetSystemBus();
|
|
|
|
DBusMessage *reply = NULL;
|
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
*nzones = 0;
|
|
|
|
*zones = NULL;
|
|
|
|
|
|
|
|
if (!sysbus)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (virDBusCallMethod(sysbus,
|
|
|
|
&reply,
|
|
|
|
NULL,
|
|
|
|
VIR_FIREWALL_FIREWALLD_SERVICE,
|
|
|
|
"/org/fedoraproject/FirewallD1",
|
|
|
|
"org.fedoraproject.FirewallD1.zone",
|
|
|
|
"getZones",
|
|
|
|
NULL) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (virDBusMessageRead(reply, "a&s", nzones, zones) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
|
|
|
virDBusMessageUnref(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* virFirewallDZoneExists:
|
|
|
|
* @match: name of zone to look for
|
|
|
|
*
|
|
|
|
* Returns true if the requested zone exists, or false if it doesn't exist
|
|
|
|
*/
|
|
|
|
bool
|
|
|
|
virFirewallDZoneExists(const char *match)
|
|
|
|
{
|
|
|
|
size_t nzones = 0, i;
|
|
|
|
char **zones = NULL;
|
|
|
|
bool result = false;
|
|
|
|
|
|
|
|
if (virFirewallDGetZones(&zones, &nzones) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
for (i = 0; i < nzones; i++) {
|
|
|
|
if (STREQ_NULLABLE(zones[i], match))
|
|
|
|
result = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
VIR_DEBUG("Requested zone '%s' %s exist",
|
|
|
|
match, result ? "does" : "doesn't");
|
|
|
|
for (i = 0; i < nzones; i++)
|
|
|
|
VIR_FREE(zones[i]);
|
|
|
|
VIR_FREE(zones);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2019-01-09 19:11:32 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* virFirewallDApplyRule:
|
|
|
|
* @layer: which layer to apply the rule to
|
|
|
|
* @args: list of args to send to this layer's passthrough command.
|
|
|
|
* @argsLen: number of items in @args
|
|
|
|
* @ignoreErrors: true to suppress logging of errors and return success
|
|
|
|
* false to log errors and return actual status
|
|
|
|
* @output: output of the direct passthrough command, if it was successful
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
virFirewallDApplyRule(virFirewallLayer layer,
|
|
|
|
char **args, size_t argsLen,
|
|
|
|
bool ignoreErrors,
|
|
|
|
char **output)
|
|
|
|
{
|
|
|
|
const char *ipv = virFirewallLayerFirewallDTypeToString(layer);
|
|
|
|
DBusConnection *sysbus = virDBusGetSystemBus();
|
|
|
|
DBusMessage *reply = NULL;
|
|
|
|
virError error;
|
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
if (!sysbus)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
memset(&error, 0, sizeof(error));
|
|
|
|
|
|
|
|
if (!ipv) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Unknown firewall layer %d"),
|
|
|
|
layer);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virDBusCallMethod(sysbus,
|
|
|
|
&reply,
|
|
|
|
&error,
|
|
|
|
VIR_FIREWALL_FIREWALLD_SERVICE,
|
|
|
|
"/org/fedoraproject/FirewallD1",
|
|
|
|
"org.fedoraproject.FirewallD1.direct",
|
|
|
|
"passthrough",
|
|
|
|
"sa&s",
|
|
|
|
ipv,
|
|
|
|
(int)argsLen,
|
|
|
|
args) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (error.level == VIR_ERR_ERROR) {
|
|
|
|
/*
|
|
|
|
* As of firewalld-0.3.9.3-1.fc20.noarch the name and
|
|
|
|
* message fields in the error look like
|
|
|
|
*
|
|
|
|
* name="org.freedesktop.DBus.Python.dbus.exceptions.DBusException"
|
|
|
|
* message="COMMAND_FAILED: '/sbin/iptables --table filter --delete
|
|
|
|
* INPUT --in-interface virbr0 --protocol udp --destination-port 53
|
|
|
|
* --jump ACCEPT' failed: iptables: Bad rule (does a matching rule
|
|
|
|
* exist in that chain?)."
|
|
|
|
*
|
|
|
|
* We'd like to only ignore DBus errors precisely related to the failure
|
|
|
|
* of iptables/ebtables commands. A well designed DBus interface would
|
|
|
|
* return specific named exceptions not the top level generic python dbus
|
|
|
|
* exception name. With this current scheme our only option is todo a
|
|
|
|
* sub-string match for 'COMMAND_FAILED' on the message. eg like
|
|
|
|
*
|
|
|
|
* if (ignoreErrors &&
|
|
|
|
* STREQ(error.name,
|
|
|
|
* "org.freedesktop.DBus.Python.dbus.exceptions.DBusException") &&
|
|
|
|
* STRPREFIX(error.message, "COMMAND_FAILED"))
|
|
|
|
* ...
|
|
|
|
*
|
|
|
|
* But this risks our error detecting code being broken if firewalld changes
|
|
|
|
* ever alter the message string, so we're avoiding doing that.
|
|
|
|
*/
|
|
|
|
if (ignoreErrors) {
|
|
|
|
VIR_DEBUG("Ignoring error '%s': '%s'",
|
|
|
|
error.str1, error.message);
|
|
|
|
} else {
|
|
|
|
virReportErrorObject(&error);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (virDBusMessageRead(reply, "s", output) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
virResetError(&error);
|
|
|
|
virDBusMessageUnref(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
2019-01-09 19:40:51 +00:00
|
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
virFirewallDInterfaceSetZone(const char *iface,
|
|
|
|
const char *zone)
|
|
|
|
{
|
|
|
|
DBusConnection *sysbus = virDBusGetSystemBus();
|
|
|
|
|
|
|
|
if (!sysbus)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return virDBusCallMethod(sysbus,
|
2019-02-13 15:57:57 +00:00
|
|
|
NULL,
|
2019-01-09 19:40:51 +00:00
|
|
|
NULL,
|
|
|
|
VIR_FIREWALL_FIREWALLD_SERVICE,
|
|
|
|
"/org/fedoraproject/FirewallD1",
|
|
|
|
"org.fedoraproject.FirewallD1.zone",
|
|
|
|
"changeZoneOfInterface",
|
|
|
|
"ss",
|
|
|
|
zone,
|
|
|
|
iface);
|
|
|
|
}
|