Blob Blame History Raw
// SPDX-License-Identifier: GPL-2.0+
/*
 * Copyright (C) 2011 Red Hat, Inc.
 */

#include "nm-default.h"

#include <arpa/inet.h>
#include <stdlib.h>

#include "nm-dispatcher-utils.h"
#include "nm-libnm-core-aux/nm-dispatcher-api.h"

#include "nm-utils/nm-test-utils.h"

#include "nmdbus-dispatcher.h"

#define TEST_DIR      NM_BUILD_SRCDIR"/dispatcher/tests"

/*****************************************************************************/

static void
_print_env (const char *const*denv, GHashTable *expected_env)
{
	const char *const*iter;
	GHashTableIter k;
	const char *key;

	g_print ("\n******* Generated environment:\n");
	for (iter = denv; iter && *iter; iter++)
		g_print ("   %s\n", *iter);

	g_print ("\n******* Expected environment:\n");
	g_hash_table_iter_init (&k, expected_env);
	while (g_hash_table_iter_next (&k, (gpointer) &key, NULL))
		g_print ("   %s\n", key);
}

static gboolean
parse_main (GKeyFile *kf,
            const char *filename,
            GVariant **out_con_dict,
            GVariant **out_con_props,
            char **out_expected_iface,
            char **out_action,
            char **out_connectivity_state,
            char **out_vpn_ip_iface,
            GError **error)
{
	nm_auto_clear_variant_builder GVariantBuilder props = { };
	gs_free char *uuid = NULL;
	gs_free char *id = NULL;
	gs_unref_object NMConnection *connection = NULL;
	NMSettingConnection *s_con;

	*out_expected_iface = g_key_file_get_string (kf, "main", "expected-iface", NULL);

	*out_connectivity_state = g_key_file_get_string (kf, "main", "connectivity-state", NULL);
	*out_vpn_ip_iface = g_key_file_get_string (kf, "main", "vpn-ip-iface", NULL);

	*out_action = g_key_file_get_string (kf, "main", "action", error);
	if (*out_action == NULL)
		return FALSE;

	uuid = g_key_file_get_string (kf, "main", "uuid", error);
	if (uuid == NULL)
		return FALSE;
	id = g_key_file_get_string (kf, "main", "id", error);
	if (id == NULL)
		return FALSE;

	connection = nm_simple_connection_new ();
	s_con = (NMSettingConnection *) nm_setting_connection_new ();
	g_object_set (s_con,
	              NM_SETTING_CONNECTION_UUID, uuid,
	              NM_SETTING_CONNECTION_ID, id,
	              NULL);
	nm_connection_add_setting (connection, NM_SETTING (s_con));

	*out_con_dict = nm_connection_to_dbus (connection, NM_CONNECTION_SERIALIZE_ALL);

	g_variant_builder_init (&props, G_VARIANT_TYPE ("a{sv}"));
	g_variant_builder_add (&props, "{sv}",
	                       NMD_CONNECTION_PROPS_PATH,
	                       g_variant_new_object_path ("/org/freedesktop/NetworkManager/Connections/5"));

	/* Strip out the non-fixed portion of the filename */
	filename = strstr (filename, "/dispatcher");
	g_variant_builder_add (&props, "{sv}",
	                       "filename",
	                       g_variant_new_string (filename));

	if (g_key_file_get_boolean (kf, "main", "external", NULL)) {
		g_variant_builder_add (&props, "{sv}",
		                       "external",
		                       g_variant_new_boolean (TRUE));
	}

	*out_con_props = g_variant_builder_end (&props);

	return TRUE;
}

static gboolean
parse_device (GKeyFile *kf, GVariant **out_device_props, GError **error)
{
	nm_auto_clear_variant_builder GVariantBuilder props = { };
	gs_free char *tmp = NULL;
	int i;

	g_variant_builder_init (&props, G_VARIANT_TYPE ("a{sv}"));

	i = g_key_file_get_integer (kf, "device", "state", error);
	if (i == 0)
		return FALSE;
	g_variant_builder_add (&props, "{sv}",
	                       NMD_DEVICE_PROPS_STATE,
	                       g_variant_new_uint32 (i));

	i = g_key_file_get_integer (kf, "device", "type", error);
	if (i == 0)
		return FALSE;
	g_variant_builder_add (&props, "{sv}",
	                       NMD_DEVICE_PROPS_TYPE,
	                       g_variant_new_uint32 (i));

	tmp = g_key_file_get_string (kf, "device", "interface", error);
	if (tmp == NULL)
		return FALSE;
	g_variant_builder_add (&props, "{sv}",
	                       NMD_DEVICE_PROPS_INTERFACE,
	                       g_variant_new_string (tmp));

	nm_clear_g_free (&tmp);
	tmp = g_key_file_get_string (kf, "device", "ip-interface", error);
	if (tmp == NULL)
		return FALSE;
	g_variant_builder_add (&props, "{sv}",
	                       NMD_DEVICE_PROPS_IP_INTERFACE,
	                       g_variant_new_string (tmp));

	nm_clear_g_free (&tmp);
	tmp = g_key_file_get_string (kf, "device", "path", error);
	if (tmp == NULL)
		return FALSE;
	g_variant_builder_add (&props, "{sv}",
	                       NMD_DEVICE_PROPS_PATH,
	                       g_variant_new_object_path (tmp));

	*out_device_props = g_variant_builder_end (&props);
	return TRUE;
}

static gboolean
add_uint_array (GKeyFile *kf,
                GVariantBuilder *props,
                const char *section,
                const char *key,
                GError **error)
{
	gs_free char *tmp = NULL;
	gs_free const char **split = NULL;
	gsize i;

	tmp = g_key_file_get_string (kf, section, key, NULL);
	if (tmp == NULL)
		return TRUE;

	split = nm_utils_strsplit_set_with_empty (tmp, " ");
	if (split) {
		gs_unref_array GArray *items = NULL;

		items = g_array_sized_new (FALSE, TRUE, sizeof (guint32), NM_PTRARRAY_LEN (split));
		for (i = 0; split[i]; i++) {
			const char *s;

			s = split[i];
			g_strstrip ((char *) s);
			if (s[0]) {
				guint32 addr;

				if (inet_pton (AF_INET, s, &addr) != 1)
					g_assert_not_reached ();
				g_array_append_val (items, addr);
			}
		}
		g_variant_builder_add (props, "{sv}", key,
		                       g_variant_new_fixed_array (G_VARIANT_TYPE_UINT32,
		                                                  items->data, items->len,
		                                                  sizeof (guint32)));
	}

	return TRUE;
}

static gboolean
parse_proxy (GKeyFile *kf, GVariant **out_props, const char *section, GError **error)
{
	nm_auto_clear_variant_builder GVariantBuilder props = { };
	gs_free char *tmp = NULL;

	g_variant_builder_init (&props, G_VARIANT_TYPE ("a{sv}"));

	tmp = g_key_file_get_string (kf, section, "pac-url", error);
	if (tmp == NULL)
		return FALSE;
	g_variant_builder_add (&props, "{sv}",
	                       "pac-url",
	                       g_variant_new_string (tmp));

	nm_clear_g_free (&tmp);
	tmp = g_key_file_get_string (kf, section, "pac-script", error);
	if (tmp == NULL)
		return FALSE;
	g_variant_builder_add (&props, "{sv}",
	                       "pac-script",
	                       g_variant_new_string (tmp));

	*out_props = g_variant_builder_end (&props);
	return TRUE;
}

static gboolean
parse_ip4 (GKeyFile *kf, GVariant **out_props, const char *section, GError **error)
{
	nm_auto_clear_variant_builder GVariantBuilder props = { };
	gs_free char *tmp = NULL;
	gs_free const char **split = NULL;
	const char **iter;

	g_variant_builder_init (&props, G_VARIANT_TYPE ("a{sv}"));

	/* search domains */
	/* Use char** for domains. (DBUS_TYPE_G_ARRAY_OF_STRING of NMIP4Config
	 * becomes G_TYPE_STRV when sending the value over D-Bus)
	 */
	tmp = g_key_file_get_string (kf, section, "domains", error);
	if (tmp == NULL)
		return FALSE;
	split = nm_utils_strsplit_set_with_empty (tmp, " ");
	if (split) {
		for (iter = split; *iter; iter++)
			g_strstrip ((char *) *iter);
		g_variant_builder_add (&props, "{sv}", "domains", g_variant_new_strv ((gpointer) split, -1));
	}
	nm_clear_g_free (&split);

	if (!add_uint_array (kf, &props, "ip4", "nameservers", error))
		return FALSE;

	if (!add_uint_array (kf, &props, "ip4", "wins-servers", error))
		return FALSE;

	nm_clear_g_free (&tmp);
	tmp = g_key_file_get_string (kf, section, "addresses", error);
	if (tmp == NULL)
		return FALSE;
	split = nm_utils_strsplit_set_with_empty (tmp, ",");
	if (split) {
		gs_unref_ptrarray GPtrArray *addresses = NULL;
		const char *gateway = NULL;

		addresses = g_ptr_array_new_with_free_func ((GDestroyNotify) nm_ip_address_unref);
		for (iter = split; *iter; iter++) {
			const char *s = *iter;
			NMIPAddress *addr;
			const char *ip;
			const char *prefix;

			g_strstrip ((char *) s);
			if (s[0] == '\0')
				continue;

			ip = *iter;

			prefix = strchr (ip, '/');
			g_assert (prefix);
			((char *) (prefix++))[0] = '\0';

			if (addresses->len == 0) {
				gateway = strchr (prefix, ' ');
				g_assert (gateway);
				gateway++;
			}

			addr = nm_ip_address_new (AF_INET, ip, (guint) atoi (prefix), error);
			if (!addr)
				return FALSE;

			g_ptr_array_add (addresses, addr);
		}

		g_variant_builder_add (&props, "{sv}", "addresses",
		                       nm_utils_ip4_addresses_to_variant (addresses, gateway));
	}
	nm_clear_g_free (&split);

	nm_clear_g_free (&tmp);
	tmp = g_key_file_get_string (kf, section, "routes", NULL);
	split = nm_utils_strsplit_set_with_empty (tmp, ",");
	if (split) {
		gs_unref_ptrarray GPtrArray *routes = NULL;

		routes = g_ptr_array_new_with_free_func ((GDestroyNotify) nm_ip_route_unref);
		for (iter = split; *iter; iter++) {
			const char *s = *iter;
			NMIPRoute *route;
			const char *dest;
			const char *prefix;
			const char *next_hop;
			const char *metric;

			g_strstrip ((char *) s);
			if (s[0] == '\0')
				continue;

			dest = s;

			prefix = strchr (dest, '/');
			g_assert (prefix);
			((char *) (prefix++))[0] = '\0';

			next_hop = strchr (prefix, ' ');
			g_assert (next_hop);
			((char *) (next_hop++))[0] = '\0';

			metric = strchr (next_hop, ' ');
			g_assert (metric);
			((char *) (metric++))[0] = '\0';

			route = nm_ip_route_new (AF_INET,
			                         dest,
			                         _nm_utils_ascii_str_to_int64 (prefix, 10, 0, 32, 255),
			                         next_hop,
			                         (guint) atoi (metric),
			                         error);
			if (!route)
				return FALSE;
			g_ptr_array_add (routes, route);
		}

		g_variant_builder_add (&props, "{sv}", "routes",
		                       nm_utils_ip4_routes_to_variant (routes));
	}

	*out_props = g_variant_builder_end (&props);
	return TRUE;
}

static gboolean
parse_dhcp (GKeyFile *kf,
            const char *group_name,
            GVariant **out_props,
            GError **error)
{
	nm_auto_clear_variant_builder GVariantBuilder props = { };
	gs_strfreev char **keys = NULL;
	char **iter;

	keys = g_key_file_get_keys (kf, group_name, NULL, error);
	if (!keys)
		return FALSE;

	g_variant_builder_init (&props, G_VARIANT_TYPE ("a{sv}"));
	for (iter = keys; iter && *iter; iter++) {
		gs_free char *val = NULL;

		val = g_key_file_get_string (kf, group_name, *iter, error);
		if (!val)
			return FALSE;
		g_variant_builder_add (&props, "{sv}", *iter, g_variant_new_string (val));
	}

	*out_props = g_variant_builder_end (&props);
	return TRUE;
}

static gboolean
get_dispatcher_file (const char *file,
                     GVariant **out_con_dict,
                     GVariant **out_con_props,
                     GVariant **out_device_props,
                     GVariant **out_device_proxy_props,
                     GVariant **out_device_ip4_props,
                     GVariant **out_device_ip6_props,
                     GVariant **out_device_dhcp4_props,
                     GVariant **out_device_dhcp6_props,
                     char **out_connectivity_state,
                     char **out_vpn_ip_iface,
                     GVariant **out_vpn_proxy_props,
                     GVariant **out_vpn_ip4_props,
                     GVariant **out_vpn_ip6_props,
                     char **out_expected_iface,
                     char **out_action,
                     GHashTable **out_env,
                     GError **error)
{
	gs_unref_keyfile GKeyFile *kf = NULL;
	gs_strfreev char **keys = NULL;
	char **iter;

	g_assert (!error || !*error);
	g_assert (out_con_dict && !*out_con_dict);
	g_assert (out_con_props && !*out_con_props);
	g_assert (out_device_props && !*out_device_props);
	g_assert (out_device_proxy_props && !*out_device_proxy_props);
	g_assert (out_device_ip4_props && !*out_device_ip4_props);
	g_assert (out_device_ip6_props && !*out_device_ip6_props);
	g_assert (out_device_dhcp4_props && !*out_device_dhcp4_props);
	g_assert (out_device_dhcp6_props && !*out_device_dhcp6_props);
	g_assert (out_connectivity_state && !*out_connectivity_state);
	g_assert (out_vpn_ip_iface && !*out_vpn_ip_iface);
	g_assert (out_vpn_proxy_props && !*out_vpn_proxy_props);
	g_assert (out_vpn_ip4_props && !*out_vpn_ip4_props);
	g_assert (out_vpn_ip6_props && !*out_vpn_ip6_props);
	g_assert (out_expected_iface && !*out_expected_iface);
	g_assert (out_action && !*out_action);
	g_assert (out_env && !*out_env);

	kf = g_key_file_new ();
	if (!g_key_file_load_from_file (kf, file, G_KEY_FILE_NONE, error))
		return FALSE;

	if (!parse_main (kf,
	                 file,
	                 out_con_dict,
	                 out_con_props,
	                 out_expected_iface,
	                 out_action,
	                 out_connectivity_state,
	                 out_vpn_ip_iface,
	                 error))
		return FALSE;

	if (!parse_device (kf, out_device_props, error))
		return FALSE;

	if (g_key_file_has_group (kf, "proxy")) {
		if (!parse_proxy (kf, out_device_proxy_props, "proxy", error))
			return FALSE;
	}

	if (g_key_file_has_group (kf, "ip4")) {
		if (!parse_ip4 (kf, out_device_ip4_props, "ip4", error))
			return FALSE;
	}

	if (g_key_file_has_group (kf, "dhcp4")) {
		if (!parse_dhcp (kf, "dhcp4", out_device_dhcp4_props, error))
			return FALSE;
	}

	if (g_key_file_has_group (kf, "dhcp6")) {
		if (!parse_dhcp (kf, "dhcp6", out_device_dhcp6_props, error))
			return FALSE;
	}

	g_assert (g_key_file_has_group (kf, "env"));
	keys = g_key_file_get_keys (kf, "env", NULL, error);
	*out_env = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, NULL);
	for (iter = keys; iter && *iter; iter++) {
		gs_free char *val = NULL;

		val = g_key_file_get_string (kf, "env", *iter, error);
		if (!val)
			return FALSE;
		g_hash_table_insert (*out_env,
		                     g_strdup_printf ("%s=%s", *iter, val),
		                     GUINT_TO_POINTER (1));
	}

	return TRUE;
}

/*****************************************************************************/

static void
test_generic (const char *file, const char *override_vpn_ip_iface)
{
	gs_unref_variant GVariant *con_dict = NULL;
	gs_unref_variant GVariant *con_props = NULL;
	gs_unref_variant GVariant *device_props = NULL;
	gs_unref_variant GVariant *device_proxy_props = NULL;
	gs_unref_variant GVariant *device_ip4_props = NULL;
	gs_unref_variant GVariant *device_ip6_props = NULL;
	gs_unref_variant GVariant *device_dhcp4_props = NULL;
	gs_unref_variant GVariant *device_dhcp6_props = NULL;
	gs_free char *connectivity_change = NULL;
	gs_free char *vpn_ip_iface = NULL;
	gs_unref_variant GVariant *vpn_proxy_props = NULL;
	gs_unref_variant GVariant *vpn_ip4_props = NULL;
	gs_unref_variant GVariant *vpn_ip6_props = NULL;
	gs_free char *expected_iface = NULL;
	gs_free char *action = NULL;
	gs_free char *out_iface = NULL;
	const char *error_message = NULL;
	gs_unref_hashtable GHashTable *expected_env = NULL;
	GError *error = NULL;
	gboolean success;
	gs_free char *filename = NULL;
	gs_strfreev char **denv = NULL;
	char **iter;

	filename = g_build_filename (TEST_DIR, file, NULL);
	success = get_dispatcher_file (filename,
	                               &con_dict,
	                               &con_props,
	                               &device_props,
	                               &device_proxy_props,
	                               &device_ip4_props,
	                               &device_ip6_props,
	                               &device_dhcp4_props,
	                               &device_dhcp6_props,
	                               &connectivity_change,
	                               &vpn_ip_iface,
	                               &vpn_proxy_props,
	                               &vpn_ip4_props,
	                               &vpn_ip6_props,
	                               &expected_iface,
	                               &action,
	                               &expected_env,
	                               &error);
	nmtst_assert_success (success, error);

	/* Get the environment from the dispatcher code */
	denv = nm_dispatcher_utils_construct_envp (action,
	                                           con_dict,
	                                           con_props,
	                                           device_props,
	                                           device_proxy_props,
	                                           device_ip4_props,
	                                           device_ip6_props,
	                                           device_dhcp4_props,
	                                           device_dhcp6_props,
	                                           connectivity_change,
	                                           override_vpn_ip_iface ?: vpn_ip_iface,
	                                           vpn_proxy_props,
	                                           vpn_ip4_props,
	                                           vpn_ip6_props,
	                                           &out_iface,
	                                           &error_message);

	g_assert ((!denv && error_message) || (denv && !error_message));

	if (error_message)
		g_error ("FAILED: %s", error_message);

	if (g_strv_length (denv) != g_hash_table_size (expected_env)) {
		_print_env (NM_CAST_STRV_CC (denv), expected_env);
		g_assert_cmpint (g_strv_length (denv), ==, g_hash_table_size (expected_env));
	}

	/* Compare dispatcher generated env and expected env */
	for (iter = denv; iter && *iter; iter++) {
		gpointer foo;
		const char *i_value = *iter;

		if (strstr (i_value, "PATH=") == i_value) {
			g_assert_cmpstr (&i_value[strlen("PATH=")], ==, g_getenv ("PATH"));

			/* The path is constructed dynamically. Ignore the actual value. */
			i_value = "PATH=";
		}

		foo = g_hash_table_lookup (expected_env, i_value);
		if (!foo) {
			_print_env (NM_CAST_STRV_CC (denv), expected_env);
			g_error ("Failed to find %s in environment", i_value);
		}
	}

	g_assert_cmpstr (expected_iface, ==, out_iface);
}

/*****************************************************************************/

static void
test_up (void)
{
	test_generic ("dispatcher-up", NULL);
}

static void
test_down (void)
{
	test_generic ("dispatcher-down", NULL);
}

static void
test_vpn_up (void)
{
	test_generic ("dispatcher-vpn-up", NULL);
}

static void
test_vpn_down (void)
{
	test_generic ("dispatcher-vpn-down", NULL);
}

static void
test_external (void)
{
	test_generic ("dispatcher-external", NULL);
}

static void
test_connectivity_changed (void)
{
	/* These tests will check that the CONNECTIVITY_STATE environment
	 * variable is only defined for known states, such as 'full'. */
	test_generic ("dispatcher-connectivity-unknown", NULL);
	test_generic ("dispatcher-connectivity-full", NULL);
}

static void
test_up_empty_vpn_iface (void)
{
	/* Test that an empty VPN iface variable, like is passed through D-Bus
	 * from NM, is ignored by the dispatcher environment construction code.
	 */
	test_generic ("dispatcher-up", "");
}

/*****************************************************************************/

static void
test_gdbus_codegen (void)
{
	gs_unref_object NMDBusDispatcher *dbus_dispatcher = NULL;

	dbus_dispatcher = nmdbus_dispatcher_skeleton_new ();
	g_assert (NMDBUS_IS_DISPATCHER_SKELETON (dbus_dispatcher));
}

/*****************************************************************************/

NMTST_DEFINE ();

int
main (int argc, char **argv)
{
	nmtst_init (&argc, &argv, TRUE);

	g_test_add_func ("/dispatcher/up", test_up);
	g_test_add_func ("/dispatcher/down", test_down);
	g_test_add_func ("/dispatcher/vpn_up", test_vpn_up);
	g_test_add_func ("/dispatcher/vpn_down", test_vpn_down);
	g_test_add_func ("/dispatcher/external", test_external);
	g_test_add_func ("/dispatcher/connectivity_changed", test_connectivity_changed);

	g_test_add_func ("/dispatcher/up_empty_vpn_iface", test_up_empty_vpn_iface);

	g_test_add_func ("/dispatcher/gdbus-codegen", test_gdbus_codegen);

	return g_test_run ();
}