mirror of
https://gitlab.com/libvirt/libvirt.git
synced 2025-01-22 04:25:18 +00:00
qemu: unescape HMP commands before converting them to json
QMP commands don't need to be escaped since converting them to json also escapes special characters. When a QMP command fails, however, libvirt falls back to HMP commands. These fallback functions (qemuMonitorText*) do their own escaping, and pass the result directly to qemuMonitorHMPCommandWithFd. If the monitor is in json mode, these pre-escaped commands will be escaped again when converted to json, which can result in the wrong arguments being sent. For example, a filename test\file would be sent in json as test\\file. This prevented attaching an image file with a " or \ in its name in qemu 1.0.50, and also broke rbd attachment (which uses backslashes to escape some internal arguments.) Reported-by: Masuko Tomoya <tomoya.masuko@gmail.com> Signed-off-by: Josh Durgin <josh.durgin@dreamhost.com> Signed-off-by: Eric Blake <eblake@redhat.com>
This commit is contained in:
parent
4716138229
commit
f27f616ff8
1
.gitignore
vendored
1
.gitignore
vendored
@ -128,6 +128,7 @@
|
|||||||
/tests/openvzutilstest
|
/tests/openvzutilstest
|
||||||
/tests/qemuargv2xmltest
|
/tests/qemuargv2xmltest
|
||||||
/tests/qemuhelptest
|
/tests/qemuhelptest
|
||||||
|
/tests/qemumonitortest
|
||||||
/tests/qemuxmlnstest
|
/tests/qemuxmlnstest
|
||||||
/tests/qparamtest
|
/tests/qparamtest
|
||||||
/tests/reconnect
|
/tests/reconnect
|
||||||
|
@ -153,6 +153,49 @@ char *qemuMonitorEscapeArg(const char *in)
|
|||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
char *qemuMonitorUnescapeArg(const char *in)
|
||||||
|
{
|
||||||
|
int i, j;
|
||||||
|
char *out;
|
||||||
|
int len = strlen(in) + 1;
|
||||||
|
char next;
|
||||||
|
|
||||||
|
if (VIR_ALLOC_N(out, len) < 0)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
for (i = j = 0; i < len; ++i) {
|
||||||
|
next = in[i];
|
||||||
|
if (in[i] == '\\') {
|
||||||
|
if (len < i + 1) {
|
||||||
|
/* trailing backslash shouldn't be possible */
|
||||||
|
VIR_FREE(out);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
++i;
|
||||||
|
switch(in[i]) {
|
||||||
|
case 'r':
|
||||||
|
next = '\r';
|
||||||
|
break;
|
||||||
|
case 'n':
|
||||||
|
next = '\n';
|
||||||
|
break;
|
||||||
|
case '"':
|
||||||
|
case '\\':
|
||||||
|
next = in[i];
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
/* invalid input */
|
||||||
|
VIR_FREE(out);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
out[j++] = next;
|
||||||
|
}
|
||||||
|
out[j] = '\0';
|
||||||
|
|
||||||
|
return out;
|
||||||
|
}
|
||||||
|
|
||||||
#if DEBUG_RAW_IO
|
#if DEBUG_RAW_IO
|
||||||
# include <c-ctype.h>
|
# include <c-ctype.h>
|
||||||
static char * qemuMonitorEscapeNonPrintable(const char *text)
|
static char * qemuMonitorEscapeNonPrintable(const char *text)
|
||||||
@ -852,10 +895,26 @@ int qemuMonitorHMPCommandWithFd(qemuMonitorPtr mon,
|
|||||||
int scm_fd,
|
int scm_fd,
|
||||||
char **reply)
|
char **reply)
|
||||||
{
|
{
|
||||||
if (mon->json)
|
char *json_cmd = NULL;
|
||||||
return qemuMonitorJSONHumanCommandWithFd(mon, cmd, scm_fd, reply);
|
int ret = -1;
|
||||||
else
|
|
||||||
return qemuMonitorTextCommandWithFd(mon, cmd, scm_fd, reply);
|
if (mon->json) {
|
||||||
|
/* hack to avoid complicating each call to text monitor functions */
|
||||||
|
json_cmd = qemuMonitorUnescapeArg(cmd);
|
||||||
|
if (!json_cmd) {
|
||||||
|
VIR_DEBUG("Could not unescape command: %s", cmd);
|
||||||
|
qemuReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
||||||
|
_("Unable to unescape command"));
|
||||||
|
goto cleanup;
|
||||||
|
}
|
||||||
|
ret = qemuMonitorJSONHumanCommandWithFd(mon, json_cmd, scm_fd, reply);
|
||||||
|
} else {
|
||||||
|
ret = qemuMonitorTextCommandWithFd(mon, cmd, scm_fd, reply);
|
||||||
|
}
|
||||||
|
|
||||||
|
cleanup:
|
||||||
|
VIR_FREE(json_cmd);
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Ensure proper locking around callbacks. */
|
/* Ensure proper locking around callbacks. */
|
||||||
|
@ -128,6 +128,7 @@ struct _qemuMonitorCallbacks {
|
|||||||
|
|
||||||
|
|
||||||
char *qemuMonitorEscapeArg(const char *in);
|
char *qemuMonitorEscapeArg(const char *in);
|
||||||
|
char *qemuMonitorUnescapeArg(const char *in);
|
||||||
|
|
||||||
qemuMonitorPtr qemuMonitorOpen(virDomainObjPtr vm,
|
qemuMonitorPtr qemuMonitorOpen(virDomainObjPtr vm,
|
||||||
virDomainChrSourceDefPtr config,
|
virDomainChrSourceDefPtr config,
|
||||||
|
@ -72,6 +72,7 @@ EXTRA_DIST = \
|
|||||||
nwfilterxml2xmlout \
|
nwfilterxml2xmlout \
|
||||||
oomtrace.pl \
|
oomtrace.pl \
|
||||||
qemuhelpdata \
|
qemuhelpdata \
|
||||||
|
qemumonitortest \
|
||||||
qemuxml2argvdata \
|
qemuxml2argvdata \
|
||||||
qemuxml2xmloutdata \
|
qemuxml2xmloutdata \
|
||||||
qemuxmlnsdata \
|
qemuxmlnsdata \
|
||||||
@ -110,7 +111,8 @@ check_PROGRAMS += xml2sexprtest sexpr2xmltest \
|
|||||||
endif
|
endif
|
||||||
if WITH_QEMU
|
if WITH_QEMU
|
||||||
check_PROGRAMS += qemuxml2argvtest qemuxml2xmltest qemuxmlnstest \
|
check_PROGRAMS += qemuxml2argvtest qemuxml2xmltest qemuxmlnstest \
|
||||||
qemuargv2xmltest qemuhelptest domainsnapshotxml2xmltest
|
qemuargv2xmltest qemuhelptest domainsnapshotxml2xmltest \
|
||||||
|
qemumonitortest
|
||||||
endif
|
endif
|
||||||
|
|
||||||
if WITH_OPENVZ
|
if WITH_OPENVZ
|
||||||
@ -237,7 +239,8 @@ endif
|
|||||||
|
|
||||||
if WITH_QEMU
|
if WITH_QEMU
|
||||||
TESTS += qemuxml2argvtest qemuxml2xmltest qemuxmlnstest qemuargv2xmltest \
|
TESTS += qemuxml2argvtest qemuxml2xmltest qemuxmlnstest qemuargv2xmltest \
|
||||||
qemuhelptest domainsnapshotxml2xmltest nwfilterxml2xmltest
|
qemuhelptest domainsnapshotxml2xmltest nwfilterxml2xmltest \
|
||||||
|
qemumonitortest
|
||||||
endif
|
endif
|
||||||
|
|
||||||
if WITH_OPENVZ
|
if WITH_OPENVZ
|
||||||
@ -365,6 +368,9 @@ qemuargv2xmltest_LDADD = $(qemu_LDADDS) $(LDADDS)
|
|||||||
qemuhelptest_SOURCES = qemuhelptest.c testutils.c testutils.h
|
qemuhelptest_SOURCES = qemuhelptest.c testutils.c testutils.h
|
||||||
qemuhelptest_LDADD = $(qemu_LDADDS) $(LDADDS)
|
qemuhelptest_LDADD = $(qemu_LDADDS) $(LDADDS)
|
||||||
|
|
||||||
|
qemumonitortest_SOURCES = qemumonitortest.c testutils.c testutils.h
|
||||||
|
qemumonitortest_LDADD = $(qemu_LDADDS) $(LDADDS)
|
||||||
|
|
||||||
domainsnapshotxml2xmltest_SOURCES = \
|
domainsnapshotxml2xmltest_SOURCES = \
|
||||||
domainsnapshotxml2xmltest.c testutilsqemu.c testutilsqemu.h \
|
domainsnapshotxml2xmltest.c testutilsqemu.c testutilsqemu.h \
|
||||||
testutils.c testutils.h
|
testutils.c testutils.h
|
||||||
@ -372,7 +378,7 @@ domainsnapshotxml2xmltest_LDADD = $(qemu_LDADDS) $(LDADDS)
|
|||||||
else
|
else
|
||||||
EXTRA_DIST += qemuxml2argvtest.c qemuxml2xmltest.c qemuargv2xmltest.c \
|
EXTRA_DIST += qemuxml2argvtest.c qemuxml2xmltest.c qemuargv2xmltest.c \
|
||||||
qemuxmlnstest.c qemuhelptest.c domainsnapshotxml2xmltest.c \
|
qemuxmlnstest.c qemuhelptest.c domainsnapshotxml2xmltest.c \
|
||||||
testutilsqemu.c testutilsqemu.h
|
qemumonitortest.c testutilsqemu.c testutilsqemu.h
|
||||||
endif
|
endif
|
||||||
|
|
||||||
if WITH_OPENVZ
|
if WITH_OPENVZ
|
||||||
|
118
tests/qemumonitortest.c
Normal file
118
tests/qemumonitortest.c
Normal file
@ -0,0 +1,118 @@
|
|||||||
|
#include <config.h>
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
|
||||||
|
#ifdef WITH_QEMU
|
||||||
|
|
||||||
|
# include "internal.h"
|
||||||
|
# include "memory.h"
|
||||||
|
# include "testutils.h"
|
||||||
|
# include "util.h"
|
||||||
|
# include "qemu/qemu_monitor.h"
|
||||||
|
|
||||||
|
struct testEscapeString
|
||||||
|
{
|
||||||
|
const char *unescaped;
|
||||||
|
const char *escaped;
|
||||||
|
};
|
||||||
|
|
||||||
|
static struct testEscapeString escapeStrings[] = {
|
||||||
|
{ "", "" },
|
||||||
|
{ " ", " " },
|
||||||
|
{ "\\", "\\\\" },
|
||||||
|
{ "\n", "\\n" },
|
||||||
|
{ "\r", "\\r" },
|
||||||
|
{ "\"", "\\\"" },
|
||||||
|
{ "\"\"\"\\\\\n\r\\\\\n\r\"\"\"", "\\\"\\\"\\\"\\\\\\\\\\n\\r\\\\\\\\\\n\\r\\\"\\\"\\\"" },
|
||||||
|
{ "drive_add dummy file=foo\\", "drive_add dummy file=foo\\\\" },
|
||||||
|
{ "block info", "block info" },
|
||||||
|
{ "set_password \":\\\"\"", "set_password \\\":\\\\\\\"\\\"" },
|
||||||
|
};
|
||||||
|
|
||||||
|
static int testEscapeArg(const void *data ATTRIBUTE_UNUSED)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
char *escaped = NULL;
|
||||||
|
for (i = 0; i < ARRAY_CARDINALITY(escapeStrings); ++i) {
|
||||||
|
escaped = qemuMonitorEscapeArg(escapeStrings[i].unescaped);
|
||||||
|
if (!escaped) {
|
||||||
|
if (virTestGetDebug() > 0) {
|
||||||
|
fprintf(stderr, "\nUnescaped string [%s]\n",
|
||||||
|
escapeStrings[i].unescaped);
|
||||||
|
fprintf(stderr, "Expect result [%s]\n",
|
||||||
|
escapeStrings[i].escaped);
|
||||||
|
fprintf(stderr, "Actual result [(null)]\n");
|
||||||
|
}
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
if (STRNEQ(escapeStrings[i].escaped, escaped)) {
|
||||||
|
virtTestDifference(stderr, escapeStrings[i].escaped, escaped);
|
||||||
|
VIR_FREE(escaped);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
VIR_FREE(escaped);
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int testUnescapeArg(const void *data ATTRIBUTE_UNUSED)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
char *unescaped = NULL;
|
||||||
|
for (i = 0; i < ARRAY_CARDINALITY(escapeStrings); ++i) {
|
||||||
|
unescaped = qemuMonitorUnescapeArg(escapeStrings[i].escaped);
|
||||||
|
if (!unescaped) {
|
||||||
|
if (virTestGetDebug() > 0) {
|
||||||
|
fprintf(stderr, "\nEscaped string [%s]\n",
|
||||||
|
escapeStrings[i].escaped);
|
||||||
|
fprintf(stderr, "Expect result [%s]\n",
|
||||||
|
escapeStrings[i].unescaped);
|
||||||
|
fprintf(stderr, "Actual result [(null)]\n");
|
||||||
|
}
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
if (STRNEQ(escapeStrings[i].unescaped, unescaped)) {
|
||||||
|
virtTestDifference(stderr, escapeStrings[i].unescaped, unescaped);
|
||||||
|
VIR_FREE(unescaped);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
VIR_FREE(unescaped);
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
mymain(void)
|
||||||
|
{
|
||||||
|
int result = 0;
|
||||||
|
|
||||||
|
# define DO_TEST(_name) \
|
||||||
|
do { \
|
||||||
|
if (virtTestRun("qemu monitor "#_name, 1, test##_name, \
|
||||||
|
NULL) < 0) { \
|
||||||
|
result = -1; \
|
||||||
|
} \
|
||||||
|
} while (0)
|
||||||
|
|
||||||
|
DO_TEST(EscapeArg);
|
||||||
|
DO_TEST(UnescapeArg);
|
||||||
|
|
||||||
|
return result == 0 ? EXIT_SUCCESS : EXIT_FAILURE;
|
||||||
|
}
|
||||||
|
|
||||||
|
VIRT_TEST_MAIN(mymain)
|
||||||
|
|
||||||
|
#else
|
||||||
|
# include "testutils.h"
|
||||||
|
|
||||||
|
int main(void)
|
||||||
|
{
|
||||||
|
return EXIT_AM_SKIP;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif /* WITH_QEMU */
|
Loading…
x
Reference in New Issue
Block a user