mirror of
https://gitlab.com/libvirt/libvirt.git
synced 2024-12-26 23:55:23 +00:00
299e97c4e8
There was a recent report of the xen-xl converter not handling config files missing an ending newline https://www.redhat.com/archives/libvir-list/2017-October/msg01353.html Commit3cc2a9e0
fixed a similar problem when parsing content of a file but missed parsing in-memory content. But AFAICT, the better fix is to properly set the end of the content when initializing the virConfParserCtxt in virConfParse(). This commit reverts the part of3cc2a9e0
that appends a newline to files missing it, and fixes setting the end of content when initializing virConfParserCtxt. A test is also added to check parsing in-memory content missing an ending newline. Signed-off-by: Jim Fehlig <jfehlig@suse.com> Reviewed-by: Daniel P. Berrange <berrange@redhat.com>
503 lines
12 KiB
C
503 lines
12 KiB
C
/*
|
|
* virconftest.c: Test the config file API
|
|
*
|
|
* Copyright (C) 2006-2016 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 <unistd.h>
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <errno.h>
|
|
#include "virconf.h"
|
|
#include "viralloc.h"
|
|
#include "testutils.h"
|
|
|
|
|
|
#define VIR_FROM_THIS VIR_FROM_NONE
|
|
|
|
static int testConfRoundTrip(const void *opaque)
|
|
{
|
|
const char *name = opaque;
|
|
int ret = -1;
|
|
virConfPtr conf = NULL;
|
|
int len = 10000;
|
|
char *buffer = NULL;
|
|
char *srcfile = NULL;
|
|
char *dstfile = NULL;
|
|
|
|
if (virAsprintf(&srcfile, "%s/virconfdata/%s.conf",
|
|
abs_srcdir, name) < 0 ||
|
|
virAsprintf(&dstfile, "%s/virconfdata/%s.out",
|
|
abs_srcdir, name) < 0)
|
|
goto cleanup;
|
|
|
|
if (VIR_ALLOC_N_QUIET(buffer, len) < 0) {
|
|
fprintf(stderr, "out of memory\n");
|
|
goto cleanup;
|
|
}
|
|
conf = virConfReadFile(srcfile, 0);
|
|
if (conf == NULL) {
|
|
fprintf(stderr, "Failed to process %s\n", srcfile);
|
|
goto cleanup;
|
|
}
|
|
ret = virConfWriteMem(buffer, &len, conf);
|
|
if (ret < 0) {
|
|
fprintf(stderr, "Failed to serialize %s back\n", srcfile);
|
|
goto cleanup;
|
|
}
|
|
|
|
if (virTestCompareToFile(buffer, dstfile) < 0)
|
|
goto cleanup;
|
|
|
|
ret = 0;
|
|
cleanup:
|
|
VIR_FREE(srcfile);
|
|
VIR_FREE(dstfile);
|
|
VIR_FREE(buffer);
|
|
virConfFree(conf);
|
|
return ret;
|
|
}
|
|
|
|
|
|
static int testConfMemoryNoNewline(const void *opaque ATTRIBUTE_UNUSED)
|
|
{
|
|
const char *srcdata = \
|
|
"ullong = '123456789'\n" \
|
|
"string = 'foo'\n" \
|
|
"uint = 12345";
|
|
|
|
virConfPtr conf = virConfReadString(srcdata, 0);
|
|
int ret = -1;
|
|
virConfValuePtr val;
|
|
unsigned long long llvalue;
|
|
char *str = NULL;
|
|
int uintvalue;
|
|
|
|
if (!conf)
|
|
return -1;
|
|
|
|
if (!(val = virConfGetValue(conf, "ullong")))
|
|
goto cleanup;
|
|
|
|
if (val->type != VIR_CONF_STRING)
|
|
goto cleanup;
|
|
|
|
if (virStrToLong_ull(val->str, NULL, 10, &llvalue) < 0)
|
|
goto cleanup;
|
|
|
|
if (llvalue != 123456789) {
|
|
fprintf(stderr, "Expected '123' got '%llu'\n", llvalue);
|
|
goto cleanup;
|
|
}
|
|
|
|
if (virConfGetValueType(conf, "string") !=
|
|
VIR_CONF_STRING) {
|
|
fprintf(stderr, "expected a string for 'string'\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
if (virConfGetValueString(conf, "string", &str) < 0)
|
|
goto cleanup;
|
|
|
|
if (STRNEQ_NULLABLE(str, "foo")) {
|
|
fprintf(stderr, "Expected 'foo' got '%s'\n", str);
|
|
goto cleanup;
|
|
}
|
|
|
|
if (virConfGetValueType(conf, "uint") != VIR_CONF_ULLONG) {
|
|
fprintf(stderr, "expected an unsigned long for 'uint'\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
if (virConfGetValueInt(conf, "uint", &uintvalue) < 0)
|
|
goto cleanup;
|
|
|
|
if (uintvalue != 12345) {
|
|
fprintf(stderr, "Expected 12345 got %ud\n", uintvalue);
|
|
goto cleanup;
|
|
}
|
|
|
|
ret = 0;
|
|
cleanup:
|
|
VIR_FREE(str);
|
|
virConfFree(conf);
|
|
return ret;
|
|
}
|
|
|
|
|
|
static int testConfParseInt(const void *opaque ATTRIBUTE_UNUSED)
|
|
{
|
|
const char *srcdata = \
|
|
"int = -1729\n" \
|
|
"uint = 1729\n" \
|
|
"llong = -6963472309248\n" \
|
|
"ullong = 6963472309248\n" \
|
|
"size_t = 87539319\n" \
|
|
"ssize_t = -87539319\n" \
|
|
"string = \"foo\"\n";
|
|
|
|
int ret = -1;
|
|
virConfPtr conf = virConfReadString(srcdata, 0);
|
|
int iv;
|
|
unsigned int ui;
|
|
size_t s;
|
|
ssize_t ss;
|
|
long long l;
|
|
unsigned long long ul;
|
|
|
|
if (!conf)
|
|
return -1;
|
|
|
|
if (virConfGetValueType(conf, "int") !=
|
|
VIR_CONF_LLONG) {
|
|
fprintf(stderr, "expected a long for 'int'\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
if (virConfGetValueInt(conf, "int", &iv) < 0)
|
|
goto cleanup;
|
|
|
|
if (iv != -1729) {
|
|
fprintf(stderr, "Expected -1729 got %d\n", iv);
|
|
goto cleanup;
|
|
}
|
|
|
|
if (virConfGetValueInt(conf, "string", &iv) != -1) {
|
|
fprintf(stderr, "Expected error for 'string' param\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
|
|
if (virConfGetValueType(conf, "uint") !=
|
|
VIR_CONF_ULLONG) {
|
|
fprintf(stderr, "expected a unsigned long for 'uint'\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
if (virConfGetValueUInt(conf, "uint", &ui) < 0)
|
|
goto cleanup;
|
|
|
|
if (ui != 1729) {
|
|
fprintf(stderr, "Expected 1729 got %u\n", ui);
|
|
goto cleanup;
|
|
}
|
|
|
|
if (virConfGetValueUInt(conf, "string", &ui) != -1) {
|
|
fprintf(stderr, "Expected error for 'string' param\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
|
|
|
|
if (virConfGetValueType(conf, "llong") !=
|
|
VIR_CONF_LLONG) {
|
|
fprintf(stderr, "expected a long for 'llong'\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
if (virConfGetValueLLong(conf, "llong", &l) < 0)
|
|
goto cleanup;
|
|
|
|
if (l != -6963472309248) {
|
|
fprintf(stderr, "Expected -6963472309248 got %lld\n", l);
|
|
goto cleanup;
|
|
}
|
|
|
|
if (virConfGetValueLLong(conf, "string", &l) != -1) {
|
|
fprintf(stderr, "Expected error for 'string' param\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
|
|
|
|
if (virConfGetValueType(conf, "ullong") !=
|
|
VIR_CONF_ULLONG) {
|
|
fprintf(stderr, "expected a unsigned long for 'ullong'\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
if (virConfGetValueULLong(conf, "ullong", &ul) < 0)
|
|
goto cleanup;
|
|
|
|
if (ul != 6963472309248) {
|
|
fprintf(stderr, "Expected 6963472309248 got %llu\n", ul);
|
|
goto cleanup;
|
|
}
|
|
|
|
if (virConfGetValueULLong(conf, "string", &ul) != -1) {
|
|
fprintf(stderr, "Expected error for 'string' param\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
|
|
|
|
if (virConfGetValueType(conf, "size_t") !=
|
|
VIR_CONF_ULLONG) {
|
|
fprintf(stderr, "expected a unsigned long for 'size_T'\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
if (virConfGetValueSizeT(conf, "size_t", &s) < 0)
|
|
goto cleanup;
|
|
|
|
if (s != 87539319) {
|
|
fprintf(stderr, "Expected 87539319 got %zu\n", s);
|
|
goto cleanup;
|
|
}
|
|
|
|
if (virConfGetValueSizeT(conf, "string", &s) != -1) {
|
|
fprintf(stderr, "Expected error for 'string' param\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
|
|
|
|
if (virConfGetValueType(conf, "ssize_t") !=
|
|
VIR_CONF_LLONG) {
|
|
fprintf(stderr, "expected a unsigned long for 'ssize_t'\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
if (virConfGetValueSSizeT(conf, "ssize_t", &ss) < 0)
|
|
goto cleanup;
|
|
|
|
if (ss != -87539319) {
|
|
fprintf(stderr, "Expected -87539319 got %zd\n", ss);
|
|
goto cleanup;
|
|
}
|
|
|
|
if (virConfGetValueSSizeT(conf, "string", &ss) != -1) {
|
|
fprintf(stderr, "Expected error for 'string' param\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
ret = 0;
|
|
cleanup:
|
|
virConfFree(conf);
|
|
return ret;
|
|
}
|
|
|
|
static int testConfParseBool(const void *opaque ATTRIBUTE_UNUSED)
|
|
{
|
|
const char *srcdata = \
|
|
"false = 0\n" \
|
|
"true = 1\n" \
|
|
"int = 6963472309248\n" \
|
|
"string = \"foo\"\n";
|
|
|
|
int ret = -1;
|
|
virConfPtr conf = virConfReadString(srcdata, 0);
|
|
bool f = true;
|
|
bool t = false;
|
|
|
|
if (!conf)
|
|
return -1;
|
|
|
|
if (virConfGetValueType(conf, "false") !=
|
|
VIR_CONF_ULLONG) {
|
|
fprintf(stderr, "expected a long for 'false'\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
if (virConfGetValueBool(conf, "false", &f) < 0)
|
|
goto cleanup;
|
|
|
|
if (f != false) {
|
|
fprintf(stderr, "Expected 0 got %d\n", f);
|
|
goto cleanup;
|
|
}
|
|
|
|
|
|
|
|
if (virConfGetValueType(conf, "true") !=
|
|
VIR_CONF_ULLONG) {
|
|
fprintf(stderr, "expected a long for 'true'\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
if (virConfGetValueBool(conf, "true", &t) < 0)
|
|
goto cleanup;
|
|
|
|
if (t != true) {
|
|
fprintf(stderr, "Expected 1 got %d\n", t);
|
|
goto cleanup;
|
|
}
|
|
|
|
|
|
|
|
if (virConfGetValueBool(conf, "int", &t) != -1) {
|
|
fprintf(stderr, "Expected error for 'string' param\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
if (virConfGetValueBool(conf, "string", &t) != -1) {
|
|
fprintf(stderr, "Expected error for 'string' param\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
|
|
ret = 0;
|
|
cleanup:
|
|
virConfFree(conf);
|
|
return ret;
|
|
}
|
|
|
|
|
|
static int testConfParseString(const void *opaque ATTRIBUTE_UNUSED)
|
|
{
|
|
const char *srcdata = \
|
|
"int = 6963472309248\n" \
|
|
"string = \"foo\"\n";
|
|
|
|
int ret = -1;
|
|
virConfPtr conf = virConfReadString(srcdata, 0);
|
|
char *str = NULL;
|
|
|
|
if (!conf)
|
|
return -1;
|
|
|
|
if (virConfGetValueType(conf, "string") !=
|
|
VIR_CONF_STRING) {
|
|
fprintf(stderr, "expected a string for 'string'\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
if (virConfGetValueString(conf, "string", &str) < 0)
|
|
goto cleanup;
|
|
|
|
if (STRNEQ_NULLABLE(str, "foo")) {
|
|
fprintf(stderr, "Expected 'foo' got '%s'\n", str);
|
|
goto cleanup;
|
|
}
|
|
|
|
if (virConfGetValueString(conf, "int", &str) != -1) {
|
|
fprintf(stderr, "Expected error for 'int'\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
ret = 0;
|
|
cleanup:
|
|
VIR_FREE(str);
|
|
virConfFree(conf);
|
|
return ret;
|
|
}
|
|
|
|
|
|
static int testConfParseStringList(const void *opaque ATTRIBUTE_UNUSED)
|
|
{
|
|
const char *srcdata = \
|
|
"string_list = [\"foo\", \"bar\"]\n" \
|
|
"string = \"foo\"\n";
|
|
|
|
int ret = -1;
|
|
virConfPtr conf = virConfReadString(srcdata, 0);
|
|
char **str = NULL;
|
|
|
|
if (!conf)
|
|
return -1;
|
|
|
|
if (virConfGetValueType(conf, "string_list") !=
|
|
VIR_CONF_LIST) {
|
|
fprintf(stderr, "expected a list for 'string_list'\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
if (virConfGetValueStringList(conf, "string_list", false, &str) < 0)
|
|
goto cleanup;
|
|
|
|
if (virStringListLength((const char *const*)str) != 2) {
|
|
fprintf(stderr, "expected a 2 element list\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
if (STRNEQ_NULLABLE(str[0], "foo")) {
|
|
fprintf(stderr, "Expected 'foo' got '%s'\n", str[0]);
|
|
goto cleanup;
|
|
}
|
|
|
|
if (STRNEQ_NULLABLE(str[1], "bar")) {
|
|
fprintf(stderr, "Expected 'bar' got '%s'\n", str[1]);
|
|
goto cleanup;
|
|
}
|
|
|
|
|
|
if (virConfGetValueStringList(conf, "string", false, &str) != -1) {
|
|
fprintf(stderr, "Expected error for 'string'\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
if (virConfGetValueStringList(conf, "string", true, &str) < 0)
|
|
goto cleanup;
|
|
|
|
if (virStringListLength((const char *const*)str) != 1) {
|
|
fprintf(stderr, "expected a 1 element list\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
if (STRNEQ_NULLABLE(str[0], "foo")) {
|
|
fprintf(stderr, "Expected 'foo' got '%s'\n", str[0]);
|
|
goto cleanup;
|
|
}
|
|
|
|
|
|
ret = 0;
|
|
cleanup:
|
|
virStringListFree(str);
|
|
virConfFree(conf);
|
|
return ret;
|
|
}
|
|
|
|
|
|
static int
|
|
mymain(void)
|
|
{
|
|
int ret = 0;
|
|
|
|
if (virTestRun("fc4", testConfRoundTrip, "fc4") < 0)
|
|
ret = -1;
|
|
|
|
if (virTestRun("libvirtd", testConfRoundTrip, "libvirtd") < 0)
|
|
ret = -1;
|
|
|
|
if (virTestRun("no-newline", testConfRoundTrip, "no-newline") < 0)
|
|
ret = -1;
|
|
|
|
if (virTestRun("memory-no-newline", testConfMemoryNoNewline, NULL) < 0)
|
|
ret = -1;
|
|
|
|
if (virTestRun("int", testConfParseInt, NULL) < 0)
|
|
ret = -1;
|
|
|
|
if (virTestRun("bool", testConfParseBool, NULL) < 0)
|
|
ret = -1;
|
|
|
|
if (virTestRun("string", testConfParseString, NULL) < 0)
|
|
ret = -1;
|
|
|
|
if (virTestRun("string-list", testConfParseStringList, NULL) < 0)
|
|
ret = -1;
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
VIR_TEST_MAIN(mymain)
|