| #include "qemu/osdep.h" |
| #include <locale.h> |
| #include <glib/gstdio.h> |
| #include <sys/socket.h> |
| #include <sys/un.h> |
| |
| #include "../qtest/libqtest.h" |
| #include "qapi/qmp/qdict.h" |
| #include "qapi/qmp/qlist.h" |
| |
| typedef struct { |
| char *test_dir; |
| GMainLoop *loop; |
| int fd; |
| GPid pid; |
| } TestFixture; |
| |
| static int connect_qga(char *path) |
| { |
| int s, ret, len, i = 0; |
| struct sockaddr_un remote; |
| |
| s = socket(AF_UNIX, SOCK_STREAM, 0); |
| g_assert(s != -1); |
| |
| remote.sun_family = AF_UNIX; |
| do { |
| strcpy(remote.sun_path, path); |
| len = strlen(remote.sun_path) + sizeof(remote.sun_family); |
| ret = connect(s, (struct sockaddr *)&remote, len); |
| if (ret == -1) { |
| g_usleep(G_USEC_PER_SEC); |
| } |
| if (i++ == 10) { |
| close(s); |
| return -1; |
| } |
| } while (ret == -1); |
| |
| return s; |
| } |
| |
| static void qga_watch(GPid pid, gint status, gpointer user_data) |
| { |
| TestFixture *fixture = user_data; |
| |
| g_assert_cmpint(status, ==, 0); |
| g_main_loop_quit(fixture->loop); |
| } |
| |
| static void |
| fixture_setup(TestFixture *fixture, gconstpointer data, gchar **envp) |
| { |
| const gchar *extra_arg = data; |
| GError *error = NULL; |
| g_autofree char *cwd = NULL; |
| g_autofree char *path = NULL; |
| g_autofree char *cmd = NULL; |
| g_auto(GStrv) argv = NULL; |
| |
| fixture->loop = g_main_loop_new(NULL, FALSE); |
| |
| fixture->test_dir = g_strdup_printf("%s/qgatest.XXXXXX", g_get_tmp_dir()); |
| g_assert_nonnull(g_mkdtemp(fixture->test_dir)); |
| |
| path = g_build_filename(fixture->test_dir, "sock", NULL); |
| cwd = g_get_current_dir(); |
| cmd = g_strdup_printf("%s%cqga%cqemu-ga -m unix-listen -t %s -p %s %s %s", |
| cwd, G_DIR_SEPARATOR, G_DIR_SEPARATOR, |
| fixture->test_dir, path, |
| getenv("QTEST_LOG") ? "-v" : "", |
| extra_arg ?: ""); |
| g_shell_parse_argv(cmd, NULL, &argv, &error); |
| g_assert_no_error(error); |
| |
| g_spawn_async(fixture->test_dir, argv, envp, |
| G_SPAWN_SEARCH_PATH|G_SPAWN_DO_NOT_REAP_CHILD, |
| NULL, NULL, &fixture->pid, &error); |
| g_assert_no_error(error); |
| |
| g_child_watch_add(fixture->pid, qga_watch, fixture); |
| |
| fixture->fd = connect_qga(path); |
| g_assert_cmpint(fixture->fd, !=, -1); |
| } |
| |
| static void |
| fixture_tear_down(TestFixture *fixture, gconstpointer data) |
| { |
| g_autofree char *tmp = NULL; |
| |
| kill(fixture->pid, SIGTERM); |
| |
| g_main_loop_run(fixture->loop); |
| g_main_loop_unref(fixture->loop); |
| |
| g_spawn_close_pid(fixture->pid); |
| |
| tmp = g_build_filename(fixture->test_dir, "foo", NULL); |
| g_unlink(tmp); |
| g_free(tmp); |
| |
| tmp = g_build_filename(fixture->test_dir, "qga.state", NULL); |
| g_unlink(tmp); |
| g_free(tmp); |
| |
| tmp = g_build_filename(fixture->test_dir, "sock", NULL); |
| g_unlink(tmp); |
| |
| g_rmdir(fixture->test_dir); |
| g_free(fixture->test_dir); |
| close(fixture->fd); |
| } |
| |
| static void qmp_assertion_message_error(const char *domain, |
| const char *file, |
| int line, |
| const char *func, |
| const char *expr, |
| QDict *dict) |
| { |
| const char *class, *desc; |
| g_autofree char *s = NULL; |
| QDict *error; |
| |
| error = qdict_get_qdict(dict, "error"); |
| class = qdict_get_try_str(error, "class"); |
| desc = qdict_get_try_str(error, "desc"); |
| |
| s = g_strdup_printf("assertion failed %s: %s %s", expr, class, desc); |
| g_assertion_message(domain, file, line, func, s); |
| } |
| |
| #define qmp_assert_no_error(err) do { \ |
| if (qdict_haskey(err, "error")) { \ |
| qmp_assertion_message_error(G_LOG_DOMAIN, __FILE__, __LINE__, \ |
| G_STRFUNC, #err, err); \ |
| } \ |
| } while (0) |
| |
| static void test_qga_sync_delimited(gconstpointer fix) |
| { |
| const TestFixture *fixture = fix; |
| guint32 v, r = g_test_rand_int(); |
| unsigned char c; |
| g_autoptr(QDict) ret = NULL; |
| |
| qmp_fd_send_raw(fixture->fd, "\xff"); |
| qmp_fd_send(fixture->fd, |
| "{'execute': 'guest-sync-delimited'," |
| " 'arguments': {'id': %u } }", |
| r); |
| |
| /* |
| * Read and ignore garbage until resynchronized. |
| * |
| * Note that the full reset sequence would involve checking the |
| * response of guest-sync-delimited and repeating the loop if |
| * 'id' field of the response does not match the 'id' field of |
| * the request. Testing this fully would require inserting |
| * garbage in the response stream and is left as a future test |
| * to implement. |
| * |
| * TODO: The server shouldn't emit so much garbage (among other |
| * things, it loudly complains about the client's \xff being |
| * invalid JSON, even though it is a documented part of the |
| * handshake. |
| */ |
| do { |
| v = read(fixture->fd, &c, 1); |
| g_assert_cmpint(v, ==, 1); |
| } while (c != 0xff); |
| |
| ret = qmp_fd_receive(fixture->fd); |
| g_assert_nonnull(ret); |
| qmp_assert_no_error(ret); |
| |
| v = qdict_get_int(ret, "return"); |
| g_assert_cmpint(r, ==, v); |
| } |
| |
| static void test_qga_sync(gconstpointer fix) |
| { |
| const TestFixture *fixture = fix; |
| guint32 v, r = g_test_rand_int(); |
| g_autoptr(QDict) ret = NULL; |
| |
| /* |
| * TODO guest-sync is inherently limited: we cannot distinguish |
| * failure caused by reacting to garbage on the wire prior to this |
| * command, from failure of this actual command. Clients are |
| * supposed to be able to send a raw '\xff' byte to at least |
| * re-synchronize the server's parser prior to this command, but |
| * we are not in a position to test that here because (at least |
| * for now) it causes the server to issue an error message about |
| * invalid JSON. Testing of '\xff' handling is done in |
| * guest-sync-delimited instead. |
| */ |
| ret = qmp_fd(fixture->fd, |
| "{'execute': 'guest-sync', 'arguments': {'id': %u } }", |
| r); |
| |
| g_assert_nonnull(ret); |
| qmp_assert_no_error(ret); |
| |
| v = qdict_get_int(ret, "return"); |
| g_assert_cmpint(r, ==, v); |
| } |
| |
| static void test_qga_ping(gconstpointer fix) |
| { |
| const TestFixture *fixture = fix; |
| g_autoptr(QDict) ret = NULL; |
| |
| ret = qmp_fd(fixture->fd, "{'execute': 'guest-ping'}"); |
| g_assert_nonnull(ret); |
| qmp_assert_no_error(ret); |
| } |
| |
| static void test_qga_id(gconstpointer fix) |
| { |
| const TestFixture *fixture = fix; |
| g_autoptr(QDict) ret = NULL; |
| |
| ret = qmp_fd(fixture->fd, "{'execute': 'guest-ping', 'id': 1}"); |
| g_assert_nonnull(ret); |
| qmp_assert_no_error(ret); |
| g_assert_cmpint(qdict_get_int(ret, "id"), ==, 1); |
| } |
| |
| static void test_qga_invalid_oob(gconstpointer fix) |
| { |
| const TestFixture *fixture = fix; |
| QDict *ret; |
| |
| ret = qmp_fd(fixture->fd, "{'exec-oob': 'guest-ping'}"); |
| g_assert_nonnull(ret); |
| |
| qmp_expect_error_and_unref(ret, "GenericError"); |
| } |
| |
| static void test_qga_invalid_args(gconstpointer fix) |
| { |
| const TestFixture *fixture = fix; |
| g_autoptr(QDict) ret = NULL; |
| QDict *error; |
| const gchar *class, *desc; |
| |
| ret = qmp_fd(fixture->fd, "{'execute': 'guest-ping', " |
| "'arguments': {'foo': 42 }}"); |
| g_assert_nonnull(ret); |
| |
| error = qdict_get_qdict(ret, "error"); |
| class = qdict_get_try_str(error, "class"); |
| desc = qdict_get_try_str(error, "desc"); |
| |
| g_assert_cmpstr(class, ==, "GenericError"); |
| g_assert_cmpstr(desc, ==, "Parameter 'foo' is unexpected"); |
| } |
| |
| static void test_qga_invalid_cmd(gconstpointer fix) |
| { |
| const TestFixture *fixture = fix; |
| g_autoptr(QDict) ret = NULL; |
| QDict *error; |
| const gchar *class, *desc; |
| |
| ret = qmp_fd(fixture->fd, "{'execute': 'guest-invalid-cmd'}"); |
| g_assert_nonnull(ret); |
| |
| error = qdict_get_qdict(ret, "error"); |
| class = qdict_get_try_str(error, "class"); |
| desc = qdict_get_try_str(error, "desc"); |
| |
| g_assert_cmpstr(class, ==, "CommandNotFound"); |
| g_assert_cmpint(strlen(desc), >, 0); |
| } |
| |
| static void test_qga_info(gconstpointer fix) |
| { |
| const TestFixture *fixture = fix; |
| g_autoptr(QDict) ret = NULL; |
| QDict *val; |
| const gchar *version; |
| |
| ret = qmp_fd(fixture->fd, "{'execute': 'guest-info'}"); |
| g_assert_nonnull(ret); |
| qmp_assert_no_error(ret); |
| |
| val = qdict_get_qdict(ret, "return"); |
| version = qdict_get_try_str(val, "version"); |
| g_assert_cmpstr(version, ==, QEMU_VERSION); |
| } |
| |
| static void test_qga_get_vcpus(gconstpointer fix) |
| { |
| const TestFixture *fixture = fix; |
| g_autoptr(QDict) ret = NULL; |
| QList *list; |
| const QListEntry *entry; |
| |
| ret = qmp_fd(fixture->fd, "{'execute': 'guest-get-vcpus'}"); |
| g_assert_nonnull(ret); |
| qmp_assert_no_error(ret); |
| |
| /* check there is at least a cpu */ |
| list = qdict_get_qlist(ret, "return"); |
| entry = qlist_first(list); |
| g_assert(qdict_haskey(qobject_to(QDict, entry->value), "online")); |
| g_assert(qdict_haskey(qobject_to(QDict, entry->value), "logical-id")); |
| } |
| |
| static void test_qga_get_fsinfo(gconstpointer fix) |
| { |
| const TestFixture *fixture = fix; |
| g_autoptr(QDict) ret = NULL; |
| QList *list; |
| const QListEntry *entry; |
| |
| ret = qmp_fd(fixture->fd, "{'execute': 'guest-get-fsinfo'}"); |
| g_assert_nonnull(ret); |
| qmp_assert_no_error(ret); |
| |
| /* sanity-check the response if there are any filesystems */ |
| list = qdict_get_qlist(ret, "return"); |
| entry = qlist_first(list); |
| if (entry) { |
| g_assert(qdict_haskey(qobject_to(QDict, entry->value), "name")); |
| g_assert(qdict_haskey(qobject_to(QDict, entry->value), "mountpoint")); |
| g_assert(qdict_haskey(qobject_to(QDict, entry->value), "type")); |
| g_assert(qdict_haskey(qobject_to(QDict, entry->value), "disk")); |
| } |
| } |
| |
| static void test_qga_get_memory_block_info(gconstpointer fix) |
| { |
| const TestFixture *fixture = fix; |
| g_autoptr(QDict) ret = NULL; |
| QDict *val; |
| int64_t size; |
| |
| ret = qmp_fd(fixture->fd, "{'execute': 'guest-get-memory-block-info'}"); |
| g_assert_nonnull(ret); |
| |
| /* some systems might not expose memory block info in sysfs */ |
| if (!qdict_haskey(ret, "error")) { |
| /* check there is at least some memory */ |
| val = qdict_get_qdict(ret, "return"); |
| size = qdict_get_int(val, "size"); |
| g_assert_cmpint(size, >, 0); |
| } |
| } |
| |
| static void test_qga_get_memory_blocks(gconstpointer fix) |
| { |
| const TestFixture *fixture = fix; |
| g_autoptr(QDict) ret = NULL; |
| QList *list; |
| const QListEntry *entry; |
| |
| ret = qmp_fd(fixture->fd, "{'execute': 'guest-get-memory-blocks'}"); |
| g_assert_nonnull(ret); |
| |
| /* some systems might not expose memory block info in sysfs */ |
| if (!qdict_haskey(ret, "error")) { |
| list = qdict_get_qlist(ret, "return"); |
| entry = qlist_first(list); |
| /* newer versions of qga may return empty list without error */ |
| if (entry) { |
| g_assert(qdict_haskey(qobject_to(QDict, entry->value), |
| "phys-index")); |
| g_assert(qdict_haskey(qobject_to(QDict, entry->value), "online")); |
| } |
| } |
| } |
| |
| static void test_qga_network_get_interfaces(gconstpointer fix) |
| { |
| const TestFixture *fixture = fix; |
| g_autoptr(QDict) ret = NULL; |
| QList *list; |
| const QListEntry *entry; |
| |
| ret = qmp_fd(fixture->fd, "{'execute': 'guest-network-get-interfaces'}"); |
| g_assert_nonnull(ret); |
| qmp_assert_no_error(ret); |
| |
| /* check there is at least an interface */ |
| list = qdict_get_qlist(ret, "return"); |
| entry = qlist_first(list); |
| g_assert(qdict_haskey(qobject_to(QDict, entry->value), "name")); |
| } |
| |
| static void test_qga_file_ops(gconstpointer fix) |
| { |
| const TestFixture *fixture = fix; |
| const unsigned char helloworld[] = "Hello World!\n"; |
| const char *b64; |
| gchar *path, *enc; |
| unsigned char *dec; |
| QDict *ret, *val; |
| int64_t id, eof; |
| gsize count; |
| FILE *f; |
| char tmp[100]; |
| |
| /* open */ |
| ret = qmp_fd(fixture->fd, "{'execute': 'guest-file-open'," |
| " 'arguments': { 'path': 'foo', 'mode': 'w+' } }"); |
| g_assert_nonnull(ret); |
| qmp_assert_no_error(ret); |
| id = qdict_get_int(ret, "return"); |
| qobject_unref(ret); |
| |
| enc = g_base64_encode(helloworld, sizeof(helloworld)); |
| /* write */ |
| ret = qmp_fd(fixture->fd, |
| "{'execute': 'guest-file-write'," |
| " 'arguments': { 'handle': %" PRId64 ", 'buf-b64': %s } }", |
| id, enc); |
| g_assert_nonnull(ret); |
| qmp_assert_no_error(ret); |
| |
| val = qdict_get_qdict(ret, "return"); |
| count = qdict_get_int(val, "count"); |
| eof = qdict_get_bool(val, "eof"); |
| g_assert_cmpint(count, ==, sizeof(helloworld)); |
| g_assert_cmpint(eof, ==, 0); |
| qobject_unref(ret); |
| |
| /* flush */ |
| ret = qmp_fd(fixture->fd, |
| "{'execute': 'guest-file-flush'," |
| " 'arguments': {'handle': %" PRId64 "} }", |
| id); |
| qobject_unref(ret); |
| |
| /* close */ |
| ret = qmp_fd(fixture->fd, |
| "{'execute': 'guest-file-close'," |
| " 'arguments': {'handle': %" PRId64 "} }", |
| id); |
| qobject_unref(ret); |
| |
| /* check content */ |
| path = g_build_filename(fixture->test_dir, "foo", NULL); |
| f = fopen(path, "r"); |
| g_free(path); |
| g_assert_nonnull(f); |
| count = fread(tmp, 1, sizeof(tmp), f); |
| g_assert_cmpint(count, ==, sizeof(helloworld)); |
| tmp[count] = 0; |
| g_assert_cmpstr(tmp, ==, (char *)helloworld); |
| fclose(f); |
| |
| /* open */ |
| ret = qmp_fd(fixture->fd, "{'execute': 'guest-file-open'," |
| " 'arguments': { 'path': 'foo', 'mode': 'r' } }"); |
| g_assert_nonnull(ret); |
| qmp_assert_no_error(ret); |
| id = qdict_get_int(ret, "return"); |
| qobject_unref(ret); |
| |
| /* read */ |
| ret = qmp_fd(fixture->fd, |
| "{'execute': 'guest-file-read'," |
| " 'arguments': { 'handle': %" PRId64 "} }", |
| id); |
| val = qdict_get_qdict(ret, "return"); |
| count = qdict_get_int(val, "count"); |
| eof = qdict_get_bool(val, "eof"); |
| b64 = qdict_get_str(val, "buf-b64"); |
| g_assert_cmpint(count, ==, sizeof(helloworld)); |
| g_assert(eof); |
| g_assert_cmpstr(b64, ==, enc); |
| |
| qobject_unref(ret); |
| g_free(enc); |
| |
| /* read eof */ |
| ret = qmp_fd(fixture->fd, |
| "{'execute': 'guest-file-read'," |
| " 'arguments': { 'handle': %" PRId64 "} }", |
| id); |
| val = qdict_get_qdict(ret, "return"); |
| count = qdict_get_int(val, "count"); |
| eof = qdict_get_bool(val, "eof"); |
| b64 = qdict_get_str(val, "buf-b64"); |
| g_assert_cmpint(count, ==, 0); |
| g_assert(eof); |
| g_assert_cmpstr(b64, ==, ""); |
| qobject_unref(ret); |
| |
| /* seek */ |
| ret = qmp_fd(fixture->fd, |
| "{'execute': 'guest-file-seek'," |
| " 'arguments': { 'handle': %" PRId64 ", " |
| " 'offset': %d, 'whence': %s } }", |
| id, 6, "set"); |
| qmp_assert_no_error(ret); |
| val = qdict_get_qdict(ret, "return"); |
| count = qdict_get_int(val, "position"); |
| eof = qdict_get_bool(val, "eof"); |
| g_assert_cmpint(count, ==, 6); |
| g_assert(!eof); |
| qobject_unref(ret); |
| |
| /* partial read */ |
| ret = qmp_fd(fixture->fd, |
| "{'execute': 'guest-file-read'," |
| " 'arguments': { 'handle': %" PRId64 "} }", |
| id); |
| val = qdict_get_qdict(ret, "return"); |
| count = qdict_get_int(val, "count"); |
| eof = qdict_get_bool(val, "eof"); |
| b64 = qdict_get_str(val, "buf-b64"); |
| g_assert_cmpint(count, ==, sizeof(helloworld) - 6); |
| g_assert(eof); |
| dec = g_base64_decode(b64, &count); |
| g_assert_cmpint(count, ==, sizeof(helloworld) - 6); |
| g_assert_cmpmem(dec, count, helloworld + 6, sizeof(helloworld) - 6); |
| g_free(dec); |
| |
| qobject_unref(ret); |
| |
| /* close */ |
| ret = qmp_fd(fixture->fd, |
| "{'execute': 'guest-file-close'," |
| " 'arguments': {'handle': %" PRId64 "} }", |
| id); |
| qobject_unref(ret); |
| } |
| |
| static void test_qga_file_write_read(gconstpointer fix) |
| { |
| const TestFixture *fixture = fix; |
| const unsigned char helloworld[] = "Hello World!\n"; |
| const char *b64; |
| gchar *enc; |
| QDict *ret, *val; |
| int64_t id, eof; |
| gsize count; |
| |
| /* open */ |
| ret = qmp_fd(fixture->fd, "{'execute': 'guest-file-open'," |
| " 'arguments': { 'path': 'foo', 'mode': 'w+' } }"); |
| g_assert_nonnull(ret); |
| qmp_assert_no_error(ret); |
| id = qdict_get_int(ret, "return"); |
| qobject_unref(ret); |
| |
| enc = g_base64_encode(helloworld, sizeof(helloworld)); |
| /* write */ |
| ret = qmp_fd(fixture->fd, |
| "{'execute': 'guest-file-write'," |
| " 'arguments': { 'handle': %" PRId64 "," |
| " 'buf-b64': %s } }", id, enc); |
| g_assert_nonnull(ret); |
| qmp_assert_no_error(ret); |
| |
| val = qdict_get_qdict(ret, "return"); |
| count = qdict_get_int(val, "count"); |
| eof = qdict_get_bool(val, "eof"); |
| g_assert_cmpint(count, ==, sizeof(helloworld)); |
| g_assert_cmpint(eof, ==, 0); |
| qobject_unref(ret); |
| |
| /* read (check implicit flush) */ |
| ret = qmp_fd(fixture->fd, |
| "{'execute': 'guest-file-read'," |
| " 'arguments': { 'handle': %" PRId64 "} }", |
| id); |
| val = qdict_get_qdict(ret, "return"); |
| count = qdict_get_int(val, "count"); |
| eof = qdict_get_bool(val, "eof"); |
| b64 = qdict_get_str(val, "buf-b64"); |
| g_assert_cmpint(count, ==, 0); |
| g_assert(eof); |
| g_assert_cmpstr(b64, ==, ""); |
| qobject_unref(ret); |
| |
| /* seek to 0 */ |
| ret = qmp_fd(fixture->fd, |
| "{'execute': 'guest-file-seek'," |
| " 'arguments': { 'handle': %" PRId64 ", " |
| " 'offset': %d, 'whence': %s } }", |
| id, 0, "set"); |
| qmp_assert_no_error(ret); |
| val = qdict_get_qdict(ret, "return"); |
| count = qdict_get_int(val, "position"); |
| eof = qdict_get_bool(val, "eof"); |
| g_assert_cmpint(count, ==, 0); |
| g_assert(!eof); |
| qobject_unref(ret); |
| |
| /* read */ |
| ret = qmp_fd(fixture->fd, |
| "{'execute': 'guest-file-read'," |
| " 'arguments': { 'handle': %" PRId64 "} }", |
| id); |
| val = qdict_get_qdict(ret, "return"); |
| count = qdict_get_int(val, "count"); |
| eof = qdict_get_bool(val, "eof"); |
| b64 = qdict_get_str(val, "buf-b64"); |
| g_assert_cmpint(count, ==, sizeof(helloworld)); |
| g_assert(eof); |
| g_assert_cmpstr(b64, ==, enc); |
| qobject_unref(ret); |
| g_free(enc); |
| |
| /* close */ |
| ret = qmp_fd(fixture->fd, |
| "{'execute': 'guest-file-close'," |
| " 'arguments': {'handle': %" PRId64 "} }", |
| id); |
| qobject_unref(ret); |
| } |
| |
| static void test_qga_get_time(gconstpointer fix) |
| { |
| const TestFixture *fixture = fix; |
| g_autoptr(QDict) ret = NULL; |
| int64_t time; |
| |
| ret = qmp_fd(fixture->fd, "{'execute': 'guest-get-time'}"); |
| g_assert_nonnull(ret); |
| qmp_assert_no_error(ret); |
| |
| time = qdict_get_int(ret, "return"); |
| g_assert_cmpint(time, >, 0); |
| } |
| |
| static void test_qga_blockedrpcs(gconstpointer data) |
| { |
| TestFixture fix; |
| QDict *ret, *error; |
| const gchar *class, *desc; |
| |
| fixture_setup(&fix, "-b guest-ping,guest-get-time", NULL); |
| |
| /* check blocked RPCs */ |
| ret = qmp_fd(fix.fd, "{'execute': 'guest-ping'}"); |
| g_assert_nonnull(ret); |
| error = qdict_get_qdict(ret, "error"); |
| class = qdict_get_try_str(error, "class"); |
| desc = qdict_get_try_str(error, "desc"); |
| g_assert_cmpstr(class, ==, "CommandNotFound"); |
| g_assert_nonnull(g_strstr_len(desc, -1, "has been disabled")); |
| qobject_unref(ret); |
| |
| ret = qmp_fd(fix.fd, "{'execute': 'guest-get-time'}"); |
| g_assert_nonnull(ret); |
| error = qdict_get_qdict(ret, "error"); |
| class = qdict_get_try_str(error, "class"); |
| desc = qdict_get_try_str(error, "desc"); |
| g_assert_cmpstr(class, ==, "CommandNotFound"); |
| g_assert_nonnull(g_strstr_len(desc, -1, "has been disabled")); |
| qobject_unref(ret); |
| |
| /* check something work */ |
| ret = qmp_fd(fix.fd, "{'execute': 'guest-get-fsinfo'}"); |
| qmp_assert_no_error(ret); |
| qobject_unref(ret); |
| |
| fixture_tear_down(&fix, NULL); |
| } |
| |
| static void test_qga_config(gconstpointer data) |
| { |
| GError *error = NULL; |
| g_autofree char *out = NULL; |
| g_autofree char *err = NULL; |
| g_autofree char *cwd = NULL; |
| g_autofree char *cmd = NULL; |
| g_auto(GStrv) argv = NULL; |
| g_auto(GStrv) strv = NULL; |
| g_autoptr(GKeyFile) kf = NULL; |
| char *str; |
| char *env[2]; |
| int status; |
| gsize n; |
| |
| cwd = g_get_current_dir(); |
| cmd = g_strdup_printf("%s%cqga%cqemu-ga -D", |
| cwd, G_DIR_SEPARATOR, G_DIR_SEPARATOR); |
| g_shell_parse_argv(cmd, NULL, &argv, &error); |
| g_assert_no_error(error); |
| |
| env[0] = g_strdup_printf("QGA_CONF=tests%cdata%ctest-qga-config", |
| G_DIR_SEPARATOR, G_DIR_SEPARATOR); |
| env[1] = NULL; |
| g_spawn_sync(NULL, argv, env, 0, |
| NULL, NULL, &out, &err, &status, &error); |
| |
| g_assert_no_error(error); |
| g_assert_cmpstr(err, ==, ""); |
| g_assert_cmpint(status, ==, 0); |
| |
| kf = g_key_file_new(); |
| g_key_file_load_from_data(kf, out, -1, G_KEY_FILE_NONE, &error); |
| g_assert_no_error(error); |
| |
| str = g_key_file_get_start_group(kf); |
| g_assert_cmpstr(str, ==, "general"); |
| g_free(str); |
| |
| g_assert_false(g_key_file_get_boolean(kf, "general", "daemon", &error)); |
| g_assert_no_error(error); |
| |
| str = g_key_file_get_string(kf, "general", "method", &error); |
| g_assert_no_error(error); |
| g_assert_cmpstr(str, ==, "virtio-serial"); |
| g_free(str); |
| |
| str = g_key_file_get_string(kf, "general", "path", &error); |
| g_assert_no_error(error); |
| g_assert_cmpstr(str, ==, "/path/to/org.qemu.guest_agent.0"); |
| g_free(str); |
| |
| str = g_key_file_get_string(kf, "general", "pidfile", &error); |
| g_assert_no_error(error); |
| g_assert_cmpstr(str, ==, "/var/foo/qemu-ga.pid"); |
| g_free(str); |
| |
| str = g_key_file_get_string(kf, "general", "statedir", &error); |
| g_assert_no_error(error); |
| g_assert_cmpstr(str, ==, "/var/state"); |
| g_free(str); |
| |
| g_assert_true(g_key_file_get_boolean(kf, "general", "verbose", &error)); |
| g_assert_no_error(error); |
| |
| strv = g_key_file_get_string_list(kf, "general", "block-rpcs", &n, &error); |
| g_assert_cmpint(n, ==, 2); |
| g_assert_true(g_strv_contains((const char * const *)strv, |
| "guest-ping")); |
| g_assert_true(g_strv_contains((const char * const *)strv, |
| "guest-get-time")); |
| g_assert_no_error(error); |
| |
| g_free(env[0]); |
| } |
| |
| static void test_qga_fsfreeze_status(gconstpointer fix) |
| { |
| const TestFixture *fixture = fix; |
| g_autoptr(QDict) ret = NULL; |
| const gchar *status; |
| |
| ret = qmp_fd(fixture->fd, "{'execute': 'guest-fsfreeze-status'}"); |
| g_assert_nonnull(ret); |
| qmp_assert_no_error(ret); |
| |
| status = qdict_get_try_str(ret, "return"); |
| g_assert_cmpstr(status, ==, "thawed"); |
| } |
| |
| static QDict *wait_for_guest_exec_completion(int fd, int64_t pid) |
| { |
| QDict *ret = NULL; |
| int64_t now; |
| bool exited; |
| QDict *val; |
| |
| now = g_get_monotonic_time(); |
| do { |
| ret = qmp_fd(fd, |
| "{'execute': 'guest-exec-status'," |
| " 'arguments': { 'pid': %" PRId64 " } }", pid); |
| g_assert_nonnull(ret); |
| val = qdict_get_qdict(ret, "return"); |
| exited = qdict_get_bool(val, "exited"); |
| if (!exited) { |
| qobject_unref(ret); |
| } |
| } while (!exited && |
| g_get_monotonic_time() < now + 5 * G_TIME_SPAN_SECOND); |
| g_assert(exited); |
| |
| return ret; |
| } |
| |
| static void test_qga_guest_exec(gconstpointer fix) |
| { |
| const TestFixture *fixture = fix; |
| g_autoptr(QDict) ret = NULL; |
| QDict *val; |
| const gchar *out; |
| g_autofree guchar *decoded = NULL; |
| int64_t pid, exitcode; |
| gsize len; |
| |
| /* exec 'echo foo bar' */ |
| ret = qmp_fd(fixture->fd, "{'execute': 'guest-exec', 'arguments': {" |
| " 'path': '/bin/echo', 'arg': [ '-n', '\" test_str \"' ]," |
| " 'capture-output': true } }"); |
| g_assert_nonnull(ret); |
| qmp_assert_no_error(ret); |
| val = qdict_get_qdict(ret, "return"); |
| pid = qdict_get_int(val, "pid"); |
| g_assert_cmpint(pid, >, 0); |
| qobject_unref(ret); |
| |
| ret = wait_for_guest_exec_completion(fixture->fd, pid); |
| |
| /* check stdout */ |
| val = qdict_get_qdict(ret, "return"); |
| exitcode = qdict_get_int(val, "exitcode"); |
| g_assert_cmpint(exitcode, ==, 0); |
| out = qdict_get_str(val, "out-data"); |
| decoded = g_base64_decode(out, &len); |
| g_assert_cmpint(len, ==, 12); |
| g_assert_cmpstr((char *)decoded, ==, "\" test_str \""); |
| } |
| |
| #if defined(G_OS_WIN32) |
| static void test_qga_guest_exec_separated(gconstpointer fix) |
| { |
| } |
| static void test_qga_guest_exec_merged(gconstpointer fix) |
| { |
| const TestFixture *fixture = fix; |
| g_autoptr(QDict) ret = NULL; |
| QDict *val; |
| const gchar *class, *desc; |
| g_autofree guchar *decoded = NULL; |
| |
| /* exec 'echo foo bar' */ |
| ret = qmp_fd(fixture->fd, "{'execute': 'guest-exec', 'arguments': {" |
| " 'path': 'echo'," |
| " 'arg': [ 'execution never reaches here' ]," |
| " 'capture-output': 'merged' } }"); |
| |
| g_assert_nonnull(ret); |
| val = qdict_get_qdict(ret, "error"); |
| g_assert_nonnull(val); |
| class = qdict_get_str(val, "class"); |
| desc = qdict_get_str(val, "desc"); |
| g_assert_cmpstr(class, ==, "GenericError"); |
| g_assert_cmpint(strlen(desc), >, 0); |
| } |
| #else |
| static void test_qga_guest_exec_separated(gconstpointer fix) |
| { |
| const TestFixture *fixture = fix; |
| g_autoptr(QDict) ret = NULL; |
| QDict *val; |
| const gchar *out, *err; |
| g_autofree guchar *out_decoded = NULL; |
| g_autofree guchar *err_decoded = NULL; |
| int64_t pid, exitcode; |
| gsize len; |
| |
| /* exec 'echo foo bar' */ |
| ret = qmp_fd(fixture->fd, "{'execute': 'guest-exec', 'arguments': {" |
| " 'path': '/bin/bash'," |
| " 'arg': [ '-c', 'for i in $(seq 4); do if (( $i %% 2 )); then echo stdout; else echo stderr 1>&2; fi; done;' ]," |
| " 'capture-output': 'separated' } }"); |
| g_assert_nonnull(ret); |
| qmp_assert_no_error(ret); |
| val = qdict_get_qdict(ret, "return"); |
| pid = qdict_get_int(val, "pid"); |
| g_assert_cmpint(pid, >, 0); |
| qobject_unref(ret); |
| |
| ret = wait_for_guest_exec_completion(fixture->fd, pid); |
| |
| val = qdict_get_qdict(ret, "return"); |
| exitcode = qdict_get_int(val, "exitcode"); |
| g_assert_cmpint(exitcode, ==, 0); |
| |
| /* check stdout */ |
| out = qdict_get_str(val, "out-data"); |
| out_decoded = g_base64_decode(out, &len); |
| g_assert_cmpint(len, ==, 14); |
| g_assert_cmpstr((char *)out_decoded, ==, "stdout\nstdout\n"); |
| |
| /* check stderr */ |
| err = qdict_get_try_str(val, "err-data"); |
| err_decoded = g_base64_decode(err, &len); |
| g_assert_cmpint(len, ==, 14); |
| g_assert_cmpstr((char *)err_decoded, ==, "stderr\nstderr\n"); |
| } |
| |
| static void test_qga_guest_exec_merged(gconstpointer fix) |
| { |
| const TestFixture *fixture = fix; |
| g_autoptr(QDict) ret = NULL; |
| QDict *val; |
| const gchar *out, *err; |
| g_autofree guchar *decoded = NULL; |
| int64_t pid, exitcode; |
| gsize len; |
| |
| /* exec 'echo foo bar' */ |
| ret = qmp_fd(fixture->fd, "{'execute': 'guest-exec', 'arguments': {" |
| " 'path': '/bin/bash'," |
| " 'arg': [ '-c', 'for i in $(seq 4); do if (( $i %% 2 )); then echo stdout; else echo stderr 1>&2; fi; done;' ]," |
| " 'capture-output': 'merged' } }"); |
| g_assert_nonnull(ret); |
| qmp_assert_no_error(ret); |
| val = qdict_get_qdict(ret, "return"); |
| pid = qdict_get_int(val, "pid"); |
| g_assert_cmpint(pid, >, 0); |
| qobject_unref(ret); |
| |
| ret = wait_for_guest_exec_completion(fixture->fd, pid); |
| |
| val = qdict_get_qdict(ret, "return"); |
| exitcode = qdict_get_int(val, "exitcode"); |
| g_assert_cmpint(exitcode, ==, 0); |
| |
| /* check stdout */ |
| out = qdict_get_str(val, "out-data"); |
| decoded = g_base64_decode(out, &len); |
| g_assert_cmpint(len, ==, 28); |
| g_assert_cmpstr((char *)decoded, ==, "stdout\nstderr\nstdout\nstderr\n"); |
| |
| /* check stderr */ |
| err = qdict_get_try_str(val, "err-data"); |
| g_assert_null(err); |
| } |
| #endif |
| |
| static void test_qga_guest_exec_invalid(gconstpointer fix) |
| { |
| const TestFixture *fixture = fix; |
| g_autoptr(QDict) ret = NULL; |
| QDict *error; |
| const gchar *class, *desc; |
| |
| /* invalid command */ |
| ret = qmp_fd(fixture->fd, "{'execute': 'guest-exec', 'arguments': {" |
| " 'path': '/bin/invalid-cmd42' } }"); |
| g_assert_nonnull(ret); |
| error = qdict_get_qdict(ret, "error"); |
| g_assert_nonnull(error); |
| class = qdict_get_str(error, "class"); |
| desc = qdict_get_str(error, "desc"); |
| g_assert_cmpstr(class, ==, "GenericError"); |
| g_assert_cmpint(strlen(desc), >, 0); |
| qobject_unref(ret); |
| |
| /* invalid pid */ |
| ret = qmp_fd(fixture->fd, "{'execute': 'guest-exec-status'," |
| " 'arguments': { 'pid': 0 } }"); |
| g_assert_nonnull(ret); |
| error = qdict_get_qdict(ret, "error"); |
| g_assert_nonnull(error); |
| class = qdict_get_str(error, "class"); |
| desc = qdict_get_str(error, "desc"); |
| g_assert_cmpstr(class, ==, "GenericError"); |
| g_assert_cmpint(strlen(desc), >, 0); |
| } |
| |
| static void test_qga_guest_get_host_name(gconstpointer fix) |
| { |
| const TestFixture *fixture = fix; |
| g_autoptr(QDict) ret = NULL; |
| QDict *val; |
| |
| ret = qmp_fd(fixture->fd, "{'execute': 'guest-get-host-name'}"); |
| g_assert_nonnull(ret); |
| qmp_assert_no_error(ret); |
| |
| val = qdict_get_qdict(ret, "return"); |
| g_assert(qdict_haskey(val, "host-name")); |
| } |
| |
| static void test_qga_guest_get_timezone(gconstpointer fix) |
| { |
| const TestFixture *fixture = fix; |
| g_autoptr(QDict) ret = NULL; |
| QDict *val; |
| |
| ret = qmp_fd(fixture->fd, "{'execute': 'guest-get-timezone'}"); |
| g_assert_nonnull(ret); |
| qmp_assert_no_error(ret); |
| |
| /* Make sure there's at least offset */ |
| val = qdict_get_qdict(ret, "return"); |
| g_assert(qdict_haskey(val, "offset")); |
| } |
| |
| static void test_qga_guest_get_users(gconstpointer fix) |
| { |
| const TestFixture *fixture = fix; |
| g_autoptr(QDict) ret = NULL; |
| QList *val; |
| |
| ret = qmp_fd(fixture->fd, "{'execute': 'guest-get-users'}"); |
| g_assert_nonnull(ret); |
| qmp_assert_no_error(ret); |
| |
| /* There is not much to test here */ |
| val = qdict_get_qlist(ret, "return"); |
| g_assert_nonnull(val); |
| } |
| |
| static void test_qga_guest_get_osinfo(gconstpointer data) |
| { |
| TestFixture fixture; |
| const gchar *str; |
| g_autoptr(QDict) ret = NULL; |
| char *env[2]; |
| QDict *val; |
| |
| env[0] = g_strdup_printf( |
| "QGA_OS_RELEASE=%s%c..%cdata%ctest-qga-os-release", |
| g_test_get_dir(G_TEST_DIST), G_DIR_SEPARATOR, G_DIR_SEPARATOR, G_DIR_SEPARATOR); |
| env[1] = NULL; |
| fixture_setup(&fixture, NULL, env); |
| |
| ret = qmp_fd(fixture.fd, "{'execute': 'guest-get-osinfo'}"); |
| g_assert_nonnull(ret); |
| qmp_assert_no_error(ret); |
| |
| val = qdict_get_qdict(ret, "return"); |
| |
| str = qdict_get_try_str(val, "id"); |
| g_assert_nonnull(str); |
| g_assert_cmpstr(str, ==, "qemu-ga-test"); |
| |
| str = qdict_get_try_str(val, "name"); |
| g_assert_nonnull(str); |
| g_assert_cmpstr(str, ==, "QEMU-GA"); |
| |
| str = qdict_get_try_str(val, "pretty-name"); |
| g_assert_nonnull(str); |
| g_assert_cmpstr(str, ==, "QEMU Guest Agent test"); |
| |
| str = qdict_get_try_str(val, "version"); |
| g_assert_nonnull(str); |
| g_assert_cmpstr(str, ==, "Test 1"); |
| |
| str = qdict_get_try_str(val, "version-id"); |
| g_assert_nonnull(str); |
| g_assert_cmpstr(str, ==, "1"); |
| |
| str = qdict_get_try_str(val, "variant"); |
| g_assert_nonnull(str); |
| g_assert_cmpstr(str, ==, "Unit test \"'$`\\ and \\\\ etc."); |
| |
| str = qdict_get_try_str(val, "variant-id"); |
| g_assert_nonnull(str); |
| g_assert_cmpstr(str, ==, "unit-test"); |
| |
| g_free(env[0]); |
| fixture_tear_down(&fixture, NULL); |
| } |
| |
| int main(int argc, char **argv) |
| { |
| TestFixture fix; |
| int ret; |
| |
| #ifdef QEMU_SANITIZE_THREAD |
| { |
| g_test_skip("tsan enabled, https://github.com/google/sanitizers/issues/1116"); |
| return 0; |
| } |
| #endif |
| |
| setlocale (LC_ALL, ""); |
| g_test_init(&argc, &argv, NULL); |
| fixture_setup(&fix, NULL, NULL); |
| |
| g_test_add_data_func("/qga/sync-delimited", &fix, test_qga_sync_delimited); |
| g_test_add_data_func("/qga/sync", &fix, test_qga_sync); |
| g_test_add_data_func("/qga/ping", &fix, test_qga_ping); |
| g_test_add_data_func("/qga/info", &fix, test_qga_info); |
| g_test_add_data_func("/qga/network-get-interfaces", &fix, |
| test_qga_network_get_interfaces); |
| if (!access("/sys/devices/system/cpu/cpu0", F_OK)) { |
| g_test_add_data_func("/qga/get-vcpus", &fix, test_qga_get_vcpus); |
| } |
| g_test_add_data_func("/qga/get-fsinfo", &fix, test_qga_get_fsinfo); |
| g_test_add_data_func("/qga/get-memory-block-info", &fix, |
| test_qga_get_memory_block_info); |
| g_test_add_data_func("/qga/get-memory-blocks", &fix, |
| test_qga_get_memory_blocks); |
| g_test_add_data_func("/qga/file-ops", &fix, test_qga_file_ops); |
| g_test_add_data_func("/qga/file-write-read", &fix, test_qga_file_write_read); |
| g_test_add_data_func("/qga/get-time", &fix, test_qga_get_time); |
| g_test_add_data_func("/qga/id", &fix, test_qga_id); |
| g_test_add_data_func("/qga/invalid-oob", &fix, test_qga_invalid_oob); |
| g_test_add_data_func("/qga/invalid-cmd", &fix, test_qga_invalid_cmd); |
| g_test_add_data_func("/qga/invalid-args", &fix, test_qga_invalid_args); |
| g_test_add_data_func("/qga/fsfreeze-status", &fix, |
| test_qga_fsfreeze_status); |
| |
| g_test_add_data_func("/qga/blockedrpcs", NULL, test_qga_blockedrpcs); |
| g_test_add_data_func("/qga/config", NULL, test_qga_config); |
| g_test_add_data_func("/qga/guest-exec", &fix, test_qga_guest_exec); |
| g_test_add_data_func("/qga/guest-exec-separated", &fix, |
| test_qga_guest_exec_separated); |
| g_test_add_data_func("/qga/guest-exec-merged", &fix, |
| test_qga_guest_exec_merged); |
| g_test_add_data_func("/qga/guest-exec-invalid", &fix, |
| test_qga_guest_exec_invalid); |
| g_test_add_data_func("/qga/guest-get-osinfo", &fix, |
| test_qga_guest_get_osinfo); |
| g_test_add_data_func("/qga/guest-get-host-name", &fix, |
| test_qga_guest_get_host_name); |
| g_test_add_data_func("/qga/guest-get-timezone", &fix, |
| test_qga_guest_get_timezone); |
| g_test_add_data_func("/qga/guest-get-users", &fix, |
| test_qga_guest_get_users); |
| |
| ret = g_test_run(); |
| |
| fixture_tear_down(&fix, NULL); |
| |
| return ret; |
| } |