mirror of
https://gitlab.com/libvirt/libvirt.git
synced 2025-01-03 05:17:54 +03:00
conf: add support for parsing/formatting virNWFilterBindingDefPtr
A typical XML representation of the virNWFilterBindingDefPtr struct looks like this: <filterbinding> <owner> <name>f25arm7</name> <uuid>12ac8b8c-4f23-4248-ae42-fdcd50c400fd</uuid> </owner> <portdev name='vnet1'/> <mac address='52:54:00:9d:81:b1'/> <filterref filter='clean-traffic'> <parameter name='MAC' value='52:54:00:9d:81:b1'/> </filterref> </filterbinding> Reviewed-by: John Ferlan <jferlan@redhat.com> Signed-off-by: Daniel P. Berrangé <berrange@redhat.com>
This commit is contained in:
parent
7c7880dd98
commit
17b1ebf4ec
@ -25,6 +25,7 @@
|
||||
#include "virstring.h"
|
||||
#include "nwfilter_params.h"
|
||||
#include "virnwfilterbindingdef.h"
|
||||
#include "viruuid.h"
|
||||
|
||||
|
||||
#define VIR_FROM_THIS VIR_FROM_NWFILTER
|
||||
@ -81,3 +82,199 @@ virNWFilterBindingDefCopy(virNWFilterBindingDefPtr src)
|
||||
virNWFilterBindingDefFree(ret);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
|
||||
static virNWFilterBindingDefPtr
|
||||
virNWFilterBindingDefParseXML(xmlXPathContextPtr ctxt)
|
||||
{
|
||||
virNWFilterBindingDefPtr ret;
|
||||
char *uuid = NULL;
|
||||
char *mac = NULL;
|
||||
xmlNodePtr node;
|
||||
|
||||
if (VIR_ALLOC(ret) < 0)
|
||||
return NULL;
|
||||
|
||||
ret->portdevname = virXPathString("string(./portdev/@name)", ctxt);
|
||||
if (!ret->portdevname) {
|
||||
virReportError(VIR_ERR_INTERNAL_ERROR,
|
||||
"%s", _("filter binding has no port dev name"));
|
||||
goto cleanup;
|
||||
}
|
||||
|
||||
if (virXPathNode("./linkdev", ctxt)) {
|
||||
ret->linkdevname = virXPathString("string(./linkdev/@name)", ctxt);
|
||||
if (!ret->linkdevname) {
|
||||
virReportError(VIR_ERR_INTERNAL_ERROR,
|
||||
"%s", _("filter binding has no link dev name"));
|
||||
goto cleanup;
|
||||
}
|
||||
}
|
||||
|
||||
ret->ownername = virXPathString("string(./owner/name)", ctxt);
|
||||
if (!ret->ownername) {
|
||||
virReportError(VIR_ERR_INTERNAL_ERROR,
|
||||
"%s", _("filter binding has no owner name"));
|
||||
goto cleanup;
|
||||
}
|
||||
|
||||
uuid = virXPathString("string(./owner/uuid)", ctxt);
|
||||
if (!uuid) {
|
||||
virReportError(VIR_ERR_INTERNAL_ERROR,
|
||||
"%s", _("filter binding has no owner UUID"));
|
||||
goto cleanup;
|
||||
}
|
||||
|
||||
if (virUUIDParse(uuid, ret->owneruuid) < 0) {
|
||||
virReportError(VIR_ERR_INTERNAL_ERROR,
|
||||
_("Unable to parse UUID '%s'"), uuid);
|
||||
VIR_FREE(uuid);
|
||||
goto cleanup;
|
||||
}
|
||||
VIR_FREE(uuid);
|
||||
|
||||
mac = virXPathString("string(./mac/@address)", ctxt);
|
||||
if (!mac) {
|
||||
virReportError(VIR_ERR_INTERNAL_ERROR,
|
||||
"%s", _("filter binding has no MAC address"));
|
||||
goto cleanup;
|
||||
}
|
||||
|
||||
if (virMacAddrParse(mac, &ret->mac) < 0) {
|
||||
virReportError(VIR_ERR_INTERNAL_ERROR,
|
||||
_("Unable to parse MAC '%s'"), mac);
|
||||
VIR_FREE(mac);
|
||||
goto cleanup;
|
||||
}
|
||||
VIR_FREE(mac);
|
||||
|
||||
ret->filter = virXPathString("string(./filterref/@filter)", ctxt);
|
||||
if (!ret->filter) {
|
||||
virReportError(VIR_ERR_INTERNAL_ERROR,
|
||||
"%s", _("filter binding has no filter reference"));
|
||||
goto cleanup;
|
||||
}
|
||||
|
||||
node = virXPathNode("./filterref", ctxt);
|
||||
if (node &&
|
||||
!(ret->filterparams = virNWFilterParseParamAttributes(node)))
|
||||
goto cleanup;
|
||||
|
||||
return ret;
|
||||
|
||||
cleanup:
|
||||
virNWFilterBindingDefFree(ret);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
|
||||
virNWFilterBindingDefPtr
|
||||
virNWFilterBindingDefParseNode(xmlDocPtr xml,
|
||||
xmlNodePtr root)
|
||||
{
|
||||
xmlXPathContextPtr ctxt = NULL;
|
||||
virNWFilterBindingDefPtr def = NULL;
|
||||
|
||||
if (STRNEQ((const char *)root->name, "filterbinding")) {
|
||||
virReportError(VIR_ERR_XML_ERROR,
|
||||
"%s",
|
||||
_("unknown root element for nwfilter binding"));
|
||||
goto cleanup;
|
||||
}
|
||||
|
||||
ctxt = xmlXPathNewContext(xml);
|
||||
if (ctxt == NULL) {
|
||||
virReportOOMError();
|
||||
goto cleanup;
|
||||
}
|
||||
|
||||
ctxt->node = root;
|
||||
def = virNWFilterBindingDefParseXML(ctxt);
|
||||
|
||||
cleanup:
|
||||
xmlXPathFreeContext(ctxt);
|
||||
return def;
|
||||
}
|
||||
|
||||
|
||||
static virNWFilterBindingDefPtr
|
||||
virNWFilterBindingDefParse(const char *xmlStr,
|
||||
const char *filename)
|
||||
{
|
||||
virNWFilterBindingDefPtr def = NULL;
|
||||
xmlDocPtr xml;
|
||||
|
||||
if ((xml = virXMLParse(filename, xmlStr, _("(nwfilterbinding_definition)")))) {
|
||||
def = virNWFilterBindingDefParseNode(xml, xmlDocGetRootElement(xml));
|
||||
xmlFreeDoc(xml);
|
||||
}
|
||||
|
||||
return def;
|
||||
}
|
||||
|
||||
|
||||
virNWFilterBindingDefPtr
|
||||
virNWFilterBindingDefParseString(const char *xmlStr)
|
||||
{
|
||||
return virNWFilterBindingDefParse(xmlStr, NULL);
|
||||
}
|
||||
|
||||
|
||||
virNWFilterBindingDefPtr
|
||||
virNWFilterBindingDefParseFile(const char *filename)
|
||||
{
|
||||
return virNWFilterBindingDefParse(NULL, filename);
|
||||
}
|
||||
|
||||
|
||||
char *
|
||||
virNWFilterBindingDefFormat(const virNWFilterBindingDef *def)
|
||||
{
|
||||
virBuffer buf = VIR_BUFFER_INITIALIZER;
|
||||
|
||||
if (virNWFilterBindingDefFormatBuf(&buf, def) < 0) {
|
||||
virBufferFreeAndReset(&buf);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
if (virBufferCheckError(&buf) < 0)
|
||||
return NULL;
|
||||
|
||||
return virBufferContentAndReset(&buf);
|
||||
}
|
||||
|
||||
|
||||
int
|
||||
virNWFilterBindingDefFormatBuf(virBufferPtr buf,
|
||||
const virNWFilterBindingDef *def)
|
||||
{
|
||||
char uuid[VIR_UUID_STRING_BUFLEN];
|
||||
char mac[VIR_MAC_STRING_BUFLEN];
|
||||
|
||||
virBufferAddLit(buf, "<filterbinding>\n");
|
||||
|
||||
virBufferAdjustIndent(buf, 2);
|
||||
|
||||
virBufferAddLit(buf, "<owner>\n");
|
||||
virBufferAdjustIndent(buf, 2);
|
||||
virBufferEscapeString(buf, "<name>%s</name>\n", def->ownername);
|
||||
virUUIDFormat(def->owneruuid, uuid);
|
||||
virBufferAsprintf(buf, "<uuid>%s</uuid>\n", uuid);
|
||||
virBufferAdjustIndent(buf, -2);
|
||||
virBufferAddLit(buf, "</owner>\n");
|
||||
|
||||
virBufferEscapeString(buf, "<portdev name='%s'/>\n", def->portdevname);
|
||||
if (def->linkdevname)
|
||||
virBufferEscapeString(buf, "<linkdev name='%s'/>\n", def->linkdevname);
|
||||
|
||||
virMacAddrFormat(&def->mac, mac);
|
||||
virBufferAsprintf(buf, "<mac address='%s'/>\n", mac);
|
||||
|
||||
if (virNWFilterFormatParamAttributes(buf, def->filterparams, def->filter) < 0)
|
||||
return -1;
|
||||
|
||||
virBufferAdjustIndent(buf, -2);
|
||||
virBufferAddLit(buf, "</filterbinding>\n");
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
@ -24,6 +24,7 @@
|
||||
# include "internal.h"
|
||||
# include "virmacaddr.h"
|
||||
# include "virhash.h"
|
||||
# include "virbuffer.h"
|
||||
|
||||
typedef struct _virNWFilterBindingDef virNWFilterBindingDef;
|
||||
typedef virNWFilterBindingDef *virNWFilterBindingDefPtr;
|
||||
@ -44,4 +45,21 @@ virNWFilterBindingDefFree(virNWFilterBindingDefPtr binding);
|
||||
virNWFilterBindingDefPtr
|
||||
virNWFilterBindingDefCopy(virNWFilterBindingDefPtr src);
|
||||
|
||||
virNWFilterBindingDefPtr
|
||||
virNWFilterBindingDefParseNode(xmlDocPtr xml,
|
||||
xmlNodePtr root);
|
||||
|
||||
virNWFilterBindingDefPtr
|
||||
virNWFilterBindingDefParseString(const char *xml);
|
||||
|
||||
virNWFilterBindingDefPtr
|
||||
virNWFilterBindingDefParseFile(const char *filename);
|
||||
|
||||
char *
|
||||
virNWFilterBindingDefFormat(const virNWFilterBindingDef *def);
|
||||
|
||||
int
|
||||
virNWFilterBindingDefFormatBuf(virBufferPtr buf,
|
||||
const virNWFilterBindingDef *def);
|
||||
|
||||
#endif /* VIR_NWFILTER_BINDING_DEF_H */
|
||||
|
@ -1049,7 +1049,12 @@ virNodeDeviceObjListRemove;
|
||||
|
||||
# conf/virnwfilterbindingdef.h
|
||||
virNWFilterBindingDefCopy;
|
||||
virNWFilterBindingDefFormat;
|
||||
virNWFilterBindingDefFormatBuf;
|
||||
virNWFilterBindingDefFree;
|
||||
virNWFilterBindingDefParseFile;
|
||||
virNWFilterBindingDefParseNode;
|
||||
virNWFilterBindingDefParseString;
|
||||
|
||||
|
||||
# conf/virnwfilterobj.h
|
||||
|
@ -157,6 +157,7 @@ EXTRA_DIST = \
|
||||
virmock.h \
|
||||
virnetdaemondata \
|
||||
virnetdevtestdata \
|
||||
virnwfilterbindingxml2xmldata \
|
||||
virpcitestdata \
|
||||
virscsidata \
|
||||
virsh-uriprecedence \
|
||||
@ -352,6 +353,7 @@ test_programs += storagebackendsheepdogtest
|
||||
endif WITH_STORAGE_SHEEPDOG
|
||||
|
||||
test_programs += nwfilterxml2xmltest
|
||||
test_programs += virnwfilterbindingxml2xmltest
|
||||
|
||||
if WITH_NWFILTER
|
||||
test_programs += nwfilterebiptablestest
|
||||
@ -848,6 +850,11 @@ nwfilterxml2xmltest_SOURCES = \
|
||||
testutils.c testutils.h
|
||||
nwfilterxml2xmltest_LDADD = $(LDADDS)
|
||||
|
||||
virnwfilterbindingxml2xmltest_SOURCES = \
|
||||
virnwfilterbindingxml2xmltest.c \
|
||||
testutils.c testutils.h
|
||||
virnwfilterbindingxml2xmltest_LDADD = $(LDADDS)
|
||||
|
||||
if WITH_NWFILTER
|
||||
nwfilterebiptablestest_SOURCES = \
|
||||
nwfilterebiptablestest.c \
|
||||
|
11
tests/virnwfilterbindingxml2xmldata/filter-vars.xml
Normal file
11
tests/virnwfilterbindingxml2xmldata/filter-vars.xml
Normal file
@ -0,0 +1,11 @@
|
||||
<filterbinding>
|
||||
<owner>
|
||||
<name>memtest</name>
|
||||
<uuid>d54df46f-1ab5-4a22-8618-4560ef5fac2c</uuid>
|
||||
</owner>
|
||||
<portdev name='vnet0'/>
|
||||
<mac address='52:54:00:7b:35:93'/>
|
||||
<filterref filter='clean-traffic'>
|
||||
<parameter name='MAC' value='52:54:00:7b:35:93'/>
|
||||
</filterref>
|
||||
</filterbinding>
|
9
tests/virnwfilterbindingxml2xmldata/simple.xml
Normal file
9
tests/virnwfilterbindingxml2xmldata/simple.xml
Normal file
@ -0,0 +1,9 @@
|
||||
<filterbinding>
|
||||
<owner>
|
||||
<name>memtest</name>
|
||||
<uuid>d54df46f-1ab5-4a22-8618-4560ef5fac2c</uuid>
|
||||
</owner>
|
||||
<portdev name='vnet0'/>
|
||||
<mac address='52:54:00:7b:35:93'/>
|
||||
<filterref filter='clean-traffic'/>
|
||||
</filterbinding>
|
112
tests/virnwfilterbindingxml2xmltest.c
Normal file
112
tests/virnwfilterbindingxml2xmltest.c
Normal file
@ -0,0 +1,112 @@
|
||||
/*
|
||||
* virnwfilterbindingxml2xmltest.c: network filter binding XML testing
|
||||
*
|
||||
* Copyright (C) 2018 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 <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <unistd.h>
|
||||
#include <string.h>
|
||||
|
||||
#include <sys/types.h>
|
||||
#include <fcntl.h>
|
||||
|
||||
#include "internal.h"
|
||||
#include "testutils.h"
|
||||
#include "virxml.h"
|
||||
#include "virnwfilterbindingdef.h"
|
||||
#include "testutilsqemu.h"
|
||||
#include "virstring.h"
|
||||
|
||||
#define VIR_FROM_THIS VIR_FROM_NONE
|
||||
|
||||
static int
|
||||
testCompareXMLToXMLFiles(const char *xml)
|
||||
{
|
||||
char *actual = NULL;
|
||||
int ret = -1;
|
||||
virNWFilterBindingDefPtr dev = NULL;
|
||||
|
||||
virResetLastError();
|
||||
|
||||
if (!(dev = virNWFilterBindingDefParseFile(xml)))
|
||||
goto fail;
|
||||
|
||||
if (!(actual = virNWFilterBindingDefFormat(dev)))
|
||||
goto fail;
|
||||
|
||||
if (virTestCompareToFile(actual, xml) < 0)
|
||||
goto fail;
|
||||
|
||||
ret = 0;
|
||||
|
||||
fail:
|
||||
VIR_FREE(actual);
|
||||
virNWFilterBindingDefFree(dev);
|
||||
return ret;
|
||||
}
|
||||
|
||||
typedef struct test_parms {
|
||||
const char *name;
|
||||
} test_parms;
|
||||
|
||||
static int
|
||||
testCompareXMLToXMLHelper(const void *data)
|
||||
{
|
||||
int result = -1;
|
||||
const test_parms *tp = data;
|
||||
char *xml = NULL;
|
||||
|
||||
if (virAsprintf(&xml, "%s/virnwfilterbindingxml2xmldata/%s.xml",
|
||||
abs_srcdir, tp->name) < 0) {
|
||||
goto cleanup;
|
||||
}
|
||||
|
||||
result = testCompareXMLToXMLFiles(xml);
|
||||
|
||||
cleanup:
|
||||
VIR_FREE(xml);
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
static int
|
||||
mymain(void)
|
||||
{
|
||||
int ret = 0;
|
||||
|
||||
#define DO_TEST(NAME) \
|
||||
do { \
|
||||
test_parms tp = { \
|
||||
.name = NAME, \
|
||||
}; \
|
||||
if (virTestRun("NWFilter XML-2-XML " NAME, \
|
||||
testCompareXMLToXMLHelper, (&tp)) < 0) \
|
||||
ret = -1; \
|
||||
} while (0)
|
||||
|
||||
DO_TEST("simple");
|
||||
DO_TEST("filter-vars");
|
||||
|
||||
return ret == 0 ? EXIT_SUCCESS : EXIT_FAILURE;
|
||||
}
|
||||
|
||||
VIR_TEST_MAIN(mymain)
|
Loading…
Reference in New Issue
Block a user