/* SPDX-License-Identifier: GPL-2.0+ */ /* * Copyright (C) 2010 - 2011 Red Hat, Inc. */ #include "nm-default.h" #include #include #include "nm-glib-aux/nm-time-utils.h" #include "nm-test-libnm-utils.h" static struct { NMTstcServiceInfo * sinfo; NMClient * client; GDBusConnection * bus; NMRemoteConnection *remote; } gl = {}; /*****************************************************************************/ static void add_cb(GObject *s, GAsyncResult *result, gpointer user_data) { gboolean *done = user_data; GError * error = NULL; gl.remote = nm_client_add_connection_finish(gl.client, result, &error); g_assert_no_error(error); *done = TRUE; g_object_add_weak_pointer(G_OBJECT(gl.remote), (void **) &gl.remote); /* nm_client_add_connection_finish() adds a ref to @remote, but we * want the weak pointer to be cleared as soon as @client drops its own ref. * So drop ours. */ g_object_unref(gl.remote); } #define TEST_CON_ID "blahblahblah" static void test_add_connection(void) { NMConnection *connection; gboolean done = FALSE; if (!nmtstc_service_available(gl.sinfo)) return; connection = nmtst_create_minimal_connection(TEST_CON_ID, NULL, NM_SETTING_WIRED_SETTING_NAME, NULL); nm_client_add_connection_async(gl.client, connection, TRUE, NULL, add_cb, &done); nmtst_main_context_iterate_until_assert(NULL, 5000, done); g_assert(gl.remote != NULL); /* Make sure the connection is the same as what we added */ g_assert( nm_connection_compare(connection, NM_CONNECTION(gl.remote), NM_SETTING_COMPARE_FLAG_EXACT) == TRUE); g_object_unref(connection); } /*****************************************************************************/ static void set_visible_cb(GObject *proxy, GAsyncResult *result, gpointer user_data) { GError * error = NULL; GVariant *ret; ret = g_dbus_proxy_call_finish(G_DBUS_PROXY(proxy), result, &error); g_assert_no_error(error); g_variant_unref(ret); } static void visible_changed_cb(GObject *object, GParamSpec *pspec, gboolean *done) { if (!nm_remote_connection_get_visible(NM_REMOTE_CONNECTION(object))) *done = TRUE; } static void connection_removed_cb(NMClient *s, NMRemoteConnection *connection, gboolean *done) { if (connection == gl.remote) *done = TRUE; } static void invis_has_settings_cb(NMSetting * setting, const char * key, const GValue *value, GParamFlags flags, gpointer user_data) { *((gboolean *) user_data) = TRUE; } static void test_make_invisible(void) { const GPtrArray *conns; int i; GDBusProxy * proxy; gboolean visible_changed = FALSE, connection_removed = FALSE; gboolean has_settings = FALSE; char * path; if (!nmtstc_service_available(gl.sinfo)) return; g_assert(gl.remote != NULL); /* Listen for the remove event when the connection becomes invisible */ g_signal_connect(gl.remote, "notify::" NM_REMOTE_CONNECTION_VISIBLE, G_CALLBACK(visible_changed_cb), &visible_changed); g_signal_connect(gl.client, "connection-removed", G_CALLBACK(connection_removed_cb), &connection_removed); path = g_strdup(nm_connection_get_path(NM_CONNECTION(gl.remote))); proxy = g_dbus_proxy_new_sync(gl.bus, G_DBUS_PROXY_FLAGS_DO_NOT_LOAD_PROPERTIES, NULL, NM_DBUS_SERVICE, path, NM_DBUS_INTERFACE_SETTINGS_CONNECTION, NULL, NULL); g_assert(proxy != NULL); /* Bypass the NMClient object so we can test it independently */ g_dbus_proxy_call(proxy, "SetVisible", g_variant_new("(b)", FALSE), G_DBUS_CALL_FLAGS_NONE, -1, NULL, set_visible_cb, NULL); /* Wait for the connection to be removed */ nmtst_main_context_iterate_until_assert(NULL, 5000, visible_changed && connection_removed); g_signal_handlers_disconnect_by_func(gl.remote, G_CALLBACK(visible_changed_cb), &visible_changed); g_signal_handlers_disconnect_by_func(gl.client, G_CALLBACK(connection_removed_cb), &connection_removed); /* Ensure NMClient no longer has the connection */ conns = nm_client_get_connections(gl.client); for (i = 0; i < conns->len; i++) { NMConnection *candidate = NM_CONNECTION(conns->pdata[i]); g_assert((gpointer) gl.remote != (gpointer) candidate); g_assert(strcmp(path, nm_connection_get_path(candidate)) != 0); } /* And ensure the invisible connection no longer has any settings */ g_assert(gl.remote); nm_connection_for_each_setting_value(NM_CONNECTION(gl.remote), invis_has_settings_cb, &has_settings); g_assert(has_settings == FALSE); g_free(path); g_object_unref(proxy); } /*****************************************************************************/ static void vis_new_connection_cb(NMClient *foo, NMRemoteConnection *connection, NMRemoteConnection **new) { *new = connection; } static void test_make_visible(void) { const GPtrArray *conns; int i; GDBusProxy * proxy; gboolean found = FALSE; char * path; NMRemoteConnection *new = NULL; if (!nmtstc_service_available(gl.sinfo)) return; g_assert(gl.remote != NULL); /* Wait for the new-connection signal when the connection is visible again */ g_signal_connect(gl.client, NM_CLIENT_CONNECTION_ADDED, G_CALLBACK(vis_new_connection_cb), &new); path = g_strdup(nm_connection_get_path(NM_CONNECTION(gl.remote))); proxy = g_dbus_proxy_new_sync(gl.bus, G_DBUS_PROXY_FLAGS_DO_NOT_LOAD_PROPERTIES, NULL, NM_DBUS_SERVICE, path, NM_DBUS_INTERFACE_SETTINGS_CONNECTION, NULL, NULL); g_assert(proxy != NULL); /* Bypass the NMClient object so we can test it independently */ g_dbus_proxy_call(proxy, "SetVisible", g_variant_new("(b)", TRUE), G_DBUS_CALL_FLAGS_NONE, -1, NULL, set_visible_cb, NULL); /* Wait for the settings service to announce the connection again */ nmtst_main_context_iterate_until_assert(NULL, 5000, new); /* Ensure the new connection is the same as the one we made visible again */ g_assert(new == gl.remote); g_signal_handlers_disconnect_by_func(gl.client, G_CALLBACK(vis_new_connection_cb), &new); /* Ensure NMClient has the connection */ conns = nm_client_get_connections(gl.client); for (i = 0; i < conns->len; i++) { NMConnection *candidate = NM_CONNECTION(conns->pdata[i]); if ((gpointer) gl.remote == (gpointer) candidate) { g_assert_cmpstr(path, ==, nm_connection_get_path(candidate)); g_assert_cmpstr(TEST_CON_ID, ==, nm_connection_get_id(candidate)); found = TRUE; break; } } g_assert(found == TRUE); g_free(path); g_object_unref(proxy); } /*****************************************************************************/ static void deleted_cb(GObject *proxy, GAsyncResult *result, gpointer user_data) { GError * error = NULL; GVariant *ret; ret = g_dbus_proxy_call_finish(G_DBUS_PROXY(proxy), result, &error); g_assert_no_error(error); g_variant_unref(ret); } static void removed_cb(NMClient *s, NMRemoteConnection *connection, gboolean *done) { if (connection == gl.remote) *done = TRUE; } static void test_remove_connection(void) { NMRemoteConnection *connection; const GPtrArray * conns; int i; GDBusProxy * proxy; gboolean done = FALSE; char * path; if (!nmtstc_service_available(gl.sinfo)) return; /* Find a connection to delete */ conns = nm_client_get_connections(gl.client); g_assert_cmpint(conns->len, >, 0); connection = NM_REMOTE_CONNECTION(conns->pdata[0]); g_assert(connection); g_assert(gl.remote == connection); path = g_strdup(nm_connection_get_path(NM_CONNECTION(connection))); g_signal_connect(gl.client, "connection-removed", G_CALLBACK(removed_cb), &done); proxy = g_dbus_proxy_new_sync(gl.bus, G_DBUS_PROXY_FLAGS_DO_NOT_LOAD_PROPERTIES, NULL, NM_DBUS_SERVICE, path, NM_DBUS_INTERFACE_SETTINGS_CONNECTION, NULL, NULL); g_assert(proxy != NULL); /* Bypass the NMClient object so we can test it independently */ g_dbus_proxy_call(proxy, "Delete", NULL, G_DBUS_CALL_FLAGS_NONE, -1, NULL, deleted_cb, NULL); nmtst_main_context_iterate_until_assert(NULL, 5000, done && !gl.remote); /* Ensure NMClient no longer has the connection */ conns = nm_client_get_connections(gl.client); for (i = 0; i < conns->len; i++) { NMConnection *candidate = NM_CONNECTION(conns->pdata[i]); g_assert((gpointer) connection != (gpointer) candidate); g_assert_cmpstr(path, ==, nm_connection_get_path(candidate)); } g_free(path); g_object_unref(proxy); } /*****************************************************************************/ #define TEST_ADD_REMOVE_ID "add-remove-test-connection" static void add_remove_cb(GObject *s, GAsyncResult *result, gpointer user_data) { NMRemoteConnection *connection; gboolean * done = user_data; gs_free_error GError *error = NULL; connection = nm_client_add_connection_finish(gl.client, result, &error); g_assert_error(error, NM_CLIENT_ERROR, NM_CLIENT_ERROR_OBJECT_CREATION_FAILED); g_assert(connection == NULL); *done = TRUE; } static void test_add_remove_connection(void) { gs_unref_variant GVariant *ret = NULL; GError * error = NULL; gs_unref_object NMConnection *connection = NULL; gboolean done = FALSE; if (!nmtstc_service_available(gl.sinfo)) return; /* This will cause the test server to immediately delete the connection * after creating it. */ ret = g_dbus_proxy_call_sync(gl.sinfo->proxy, "AutoRemoveNextConnection", NULL, G_DBUS_CALL_FLAGS_NONE, -1, NULL, &error); nmtst_assert_success(ret, error); connection = nmtst_create_minimal_connection(TEST_ADD_REMOVE_ID, NULL, NM_SETTING_WIRED_SETTING_NAME, NULL); nm_client_add_connection_async(gl.client, connection, TRUE, NULL, add_remove_cb, &done); nmtst_main_context_iterate_until_assert(NULL, 5000, done); } /*****************************************************************************/ static void add_bad_cb(GObject *s, GAsyncResult *result, gpointer user_data) { gboolean * done = user_data; gs_free_error GError *error = NULL; gl.remote = nm_client_add_connection_finish(gl.client, result, &error); g_assert_error(error, NM_CONNECTION_ERROR, NM_CONNECTION_ERROR_INVALID_PROPERTY); *done = TRUE; } static void test_add_bad_connection(void) { gs_unref_object NMConnection *connection = NULL; gboolean done = FALSE; if (!nmtstc_service_available(gl.sinfo)) return; /* The test daemon doesn't support bond connections */ connection = nmtst_create_minimal_connection("bad connection test", NULL, NM_SETTING_BOND_SETTING_NAME, NULL); nm_client_add_connection_async(gl.client, connection, TRUE, NULL, add_bad_cb, &done); g_clear_object(&connection); nmtst_main_context_iterate_until_assert(NULL, 5000, done); g_assert(gl.remote == NULL); } /*****************************************************************************/ static void save_hostname_cb(GObject *s, GAsyncResult *result, gpointer user_data) { gboolean * done = user_data; gs_free_error GError *error = NULL; nm_client_save_hostname_finish(gl.client, result, &error); g_assert_no_error(error); *done = TRUE; } static void test_save_hostname(void) { gint64 until_ts; gboolean done = FALSE; GError * error = NULL; if (!nmtstc_service_available(gl.sinfo)) return; /* test-networkmanager-service.py requires the hostname to contain a '.' */ nm_client_save_hostname(gl.client, "foo", NULL, &error); g_assert_error(error, NM_SETTINGS_ERROR, NM_SETTINGS_ERROR_INVALID_HOSTNAME); g_clear_error(&error); nm_client_save_hostname_async(gl.client, "example.com", NULL, save_hostname_cb, &done); until_ts = nm_utils_get_monotonic_timestamp_msec() + 5000; while (TRUE) { g_main_context_iteration(NULL, FALSE); if (done) break; if (nm_utils_get_monotonic_timestamp_msec() >= until_ts) g_assert_not_reached(); } g_assert(gl.remote == NULL); } /*****************************************************************************/ NMTST_DEFINE(); int main(int argc, char **argv) { int ret; GError *error = NULL; g_setenv("LIBNM_USE_SESSION_BUS", "1", TRUE); nmtst_init(&argc, &argv, TRUE); gl.bus = g_bus_get_sync(G_BUS_TYPE_SESSION, NULL, &error); nmtst_assert_success(gl.bus, error); gl.sinfo = nmtstc_service_init(); gl.client = nmtstc_client_new(TRUE); /* FIXME: these tests assume that they get run in order, but g_test_run() * does not actually guarantee that! */ g_test_add_func("/client/add_connection", test_add_connection); g_test_add_func("/client/make_invisible", test_make_invisible); g_test_add_func("/client/make_visible", test_make_visible); g_test_add_func("/client/remove_connection", test_remove_connection); g_test_add_func("/client/add_remove_connection", test_add_remove_connection); g_test_add_func("/client/add_bad_connection", test_add_bad_connection); g_test_add_func("/client/save_hostname", test_save_hostname); ret = g_test_run(); nm_clear_pointer(&gl.sinfo, nmtstc_service_cleanup); g_clear_object(&gl.client); g_clear_object(&gl.bus); return ret; }