Blame shared/nm-utils/nm-test-utils.h

Packit Service d328f3
/*
Packit Service d328f3
 * This library is free software; you can redistribute it and/or
Packit Service d328f3
 * modify it under the terms of the GNU Lesser General Public
Packit Service d328f3
 * License as published by the Free Software Foundation; either
Packit Service d328f3
 * version 2 of the License, or (at your option) any later version.
Packit Service d328f3
 *
Packit Service d328f3
 * This library is distributed in the hope that it will be useful,
Packit Service d328f3
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service d328f3
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit Service d328f3
 * Lesser General Public License for more details.
Packit Service d328f3
 *
Packit Service d328f3
 * You should have received a copy of the GNU Lesser General Public
Packit Service d328f3
 * License along with this library; if not, write to the
Packit Service d328f3
 * Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
Packit Service d328f3
 * Boston, MA 02110-1301 USA.
Packit Service d328f3
 *
Packit Service d328f3
 * Copyright 2014 Red Hat, Inc.
Packit Service d328f3
 */
Packit Service d328f3
Packit Service d328f3
#ifndef __NM_TEST_UTILS_H__
Packit Service d328f3
#define __NM_TEST_UTILS_H__
Packit Service d328f3
Packit Service d328f3
/*******************************************************************************
Packit Service d328f3
 * HOWTO run tests.
Packit Service d328f3
 *
Packit Service d328f3
 * Our tests (make check) include this header-only file nm-test-utils.h.
Packit Service d328f3
 *
Packit Service d328f3
 * Logging:
Packit Service d328f3
 *   In tests, nm-logging redirects to glib logging. By default, glib suppresses all debug
Packit Service d328f3
 *   messages unless you set G_MESSAGES_DEBUG. To enable debug logging, you can explicitly set
Packit Service d328f3
 *   G_MESSAGES_DEBUG. Otherwise, nm-test will set G_MESSAGES_DEBUG=all in debug mode (see below).
Packit Service d328f3
 *   For nm-logging, you can configure the log-level and domains via NMTST_DEBUG environment
Packit Service d328f3
 *   variable.
Packit Service d328f3
 *
Packit Service d328f3
 * Assert-logging:
Packit Service d328f3
 *   Some tests assert against logged messages (g_test_expect_message()).
Packit Service d328f3
 *   By specifying no-expect-message in NMTST_DEBUG, you can disable assert logging
Packit Service d328f3
 *   and g_test_assert_expected_messages() will not fail.
Packit Service d328f3
 *
Packit Service d328f3
 * NMTST_SEED_RAND environment variable:
Packit Service d328f3
 *   Tests that use random numbers from nmtst_get_rand() get seeded randomly at each start.
Packit Service d328f3
 *   You can specify the seed by setting NMTST_SEED_RAND. Also, tests will print the seed
Packit Service d328f3
 *   to stdout, so that you know the chosen seed.
Packit Service d328f3
 *
Packit Service d328f3
 *
Packit Service d328f3
 * NMTST_DEBUG environment variable:
Packit Service d328f3
 *
Packit Service d328f3
 * "debug", "no-debug": when at test is run in debug mode, it might behave differently,
Packit Service d328f3
 *   depending on the test. See nmtst_is_debug().
Packit Service d328f3
 *   Known differences:
Packit Service d328f3
 *    - a test might leave the logging level unspecified. In this case, running in
Packit Service d328f3
 *      debug mode, will turn on DEBUG logging, otherwise WARN logging only.
Packit Service d328f3
 *    - if G_MESSAGES_DEBUG is unset, nm-test will set G_MESSAGES_DEBUG=all
Packit Service d328f3
 *      for tests that don't do assert-logging.
Packit Service d328f3
 *   Debug mode is determined as follows (highest priority first):
Packit Service d328f3
 *    - command line option --debug/--no-debug
Packit Service d328f3
 *    - NMTST_DEBUG=debug/no-debug
Packit Service d328f3
 *    - setting NMTST_DEBUG implies debugging turned on
Packit Service d328f3
 *    - g_test_verbose()
Packit Service d328f3
 *
Packit Service d328f3
 * "no-expect-message": for tests that would assert against log messages, disable
Packit Service d328f3
 *   those asserts.
Packit Service d328f3
 *
Packit Service d328f3
 * "log-level=LEVEL", "log-domains=DOMAIN": reset the log level and domain for tests.
Packit Service d328f3
 *    It only has an effect for nm-logging messages.
Packit Service d328f3
 *    This has no effect if the test asserts against logging (unless no-expect-message),
Packit Service d328f3
 *    otherwise, changing the logging would break tests.
Packit Service d328f3
 *    If you set the level to DEBUG or TRACE, it also sets G_MESSAGES_DEBUG=all (unless
Packit Service d328f3
 *    in assert-logging mode and unless G_MESSAGES_DEBUG is already defined).
Packit Service d328f3
 *
Packit Service d328f3
 * "TRACE", this is shorthand for "log-level=TRACE".
Packit Service d328f3
 *
Packit Service d328f3
 * "D", this is shorthand for "log-level=TRACE,no-expect-message".
Packit Service d328f3
 *
Packit Service d328f3
 * "sudo-cmd=PATH": when running root tests as normal user, the test will execute
Packit Service d328f3
 *   itself by invoking sudo at PATH.
Packit Service d328f3
 *   For example
Packit Service d328f3
 *     NMTST_DEBUG="sudo-cmd=$PWD/tools/test-sudo-wrapper.sh" make -C src/platform/tests/ check
Packit Service d328f3
 *
Packit Service d328f3
 * "slow|quick|thorough": enable/disable long-running tests. This sets nmtst_test_quick().
Packit Service d328f3
 *   Whether long-running tests are enabled is determined as follows (highest priority first):
Packit Service d328f3
 *     - specifying the value in NMTST_DEBUG has highest priority
Packit Service d328f3
 *     - respect g_test_quick(), if the command line contains '-mslow', '-mquick', '-mthorough'.
Packit Service d328f3
 *     - use compile time default (CFLAGS=-DNMTST_TEST_QUICK=TRUE)
Packit Service d328f3
 *     - enable slow tests by default
Packit Service d328f3
 *
Packit Service d328f3
 * "p=PATH"|"s=PATH": passes the path to g_test_init() as "-p" and "-s", respectively.
Packit Service d328f3
 *   Unfortunately, these options conflict with "--tap" which our makefile passes to the
Packit Service d328f3
 *   tests, thus it's only useful outside of `make check`.
Packit Service d328f3
 *
Packit Service d328f3
 *******************************************************************************/
Packit Service d328f3
Packit Service d328f3
#include "nm-default.h"
Packit Service d328f3
Packit Service d328f3
#if defined(NM_ASSERT_NO_MSG) && NM_ASSERT_NO_MSG
Packit Service d328f3
#undef g_return_if_fail_warning
Packit Service d328f3
#undef g_assertion_message_expr
Packit Service d328f3
#endif
Packit Service d328f3
Packit Service d328f3
#include <arpa/inet.h>
Packit Service d328f3
#include <stdio.h>
Packit Service d328f3
#include <unistd.h>
Packit Service d328f3
#include <stdlib.h>
Packit Service d328f3
#include <string.h>
Packit Service d328f3
#include <errno.h>
Packit Service d328f3
Packit Service d328f3
#include "nm-utils.h"
Packit Service d328f3
Packit Service d328f3
/*****************************************************************************/
Packit Service d328f3
Packit Service d328f3
#define NMTST_G_RETURN_MSG_S(expr) "*: assertion '"NM_ASSERT_G_RETURN_EXPR(expr)"' failed"
Packit Service d328f3
#define NMTST_G_RETURN_MSG(expr)   NMTST_G_RETURN_MSG_S(#expr)
Packit Service d328f3
Packit Service d328f3
/*****************************************************************************/
Packit Service d328f3
Packit Service d328f3
/* general purpose functions that have no dependency on other nmtst functions */
Packit Service d328f3
Packit Service d328f3
#define nmtst_assert_error(error, expect_error_domain, expect_error_code, expect_error_pattern) \
Packit Service d328f3
	G_STMT_START { \
Packit Service d328f3
		GError *_error = (error); \
Packit Service d328f3
		GQuark _expect_error_domain = (expect_error_domain); \
Packit Service d328f3
		const char *_expect_error_pattern = (expect_error_pattern); \
Packit Service d328f3
		\
Packit Service d328f3
		if (_expect_error_domain) \
Packit Service d328f3
			g_assert_error (_error, _expect_error_domain, (expect_error_code)); \
Packit Service d328f3
		else \
Packit Service d328f3
			g_assert (_error); \
Packit Service d328f3
		g_assert (_error->message); \
Packit Service d328f3
		if (   _expect_error_pattern \
Packit Service d328f3
		    && !g_pattern_match_simple (_expect_error_pattern, _error->message)) { \
Packit Service d328f3
			g_error ("%s:%d: error message does not have expected pattern '%s'. Instead it is '%s' (%s, %d)", \
Packit Service d328f3
			         __FILE__, __LINE__, \
Packit Service d328f3
			         _expect_error_pattern, _error->message, g_quark_to_string (_error->domain), _error->code); \
Packit Service d328f3
		} \
Packit Service d328f3
	} G_STMT_END
Packit Service d328f3
Packit Service d328f3
#define NMTST_WAIT(max_wait_ms, wait) \
Packit Service d328f3
	({ \
Packit Service d328f3
		gboolean _not_expired = TRUE; \
Packit Service d328f3
		const gint64 nmtst_wait_start_us = g_get_monotonic_time (); \
Packit Service d328f3
		const gint64 nmtst_wait_duration_us = (max_wait_ms) * 1000L; \
Packit Service d328f3
		const gint64 nmtst_wait_end_us = nmtst_wait_start_us + nmtst_wait_duration_us; \
Packit Service d328f3
		\
Packit Service d328f3
		while (TRUE) { \
Packit Service d328f3
			{ wait }; \
Packit Service d328f3
			if (g_get_monotonic_time () > nmtst_wait_end_us) { \
Packit Service d328f3
				_not_expired = FALSE; \
Packit Service d328f3
				break; \
Packit Service d328f3
			} \
Packit Service d328f3
		} \
Packit Service d328f3
		_not_expired; \
Packit Service d328f3
	})
Packit Service d328f3
Packit Service d328f3
#define NMTST_WAIT_ASSERT(max_wait_ms, wait) \
Packit Service d328f3
	G_STMT_START { \
Packit Service d328f3
		if (!(NMTST_WAIT (max_wait_ms, wait))) \
Packit Service d328f3
			g_assert_not_reached (); \
Packit Service d328f3
	} G_STMT_END
Packit Service d328f3
Packit Service d328f3
#define nmtst_assert_success(success, error) \
Packit Service d328f3
	G_STMT_START { \
Packit Service d328f3
		g_assert_no_error (error); \
Packit Service d328f3
		g_assert ((success)); \
Packit Service d328f3
	} G_STMT_END
Packit Service d328f3
Packit Service d328f3
#define nmtst_assert_no_success(success, error) \
Packit Service d328f3
	G_STMT_START { \
Packit Service d328f3
		g_assert (error); \
Packit Service d328f3
		g_assert (!(success)); \
Packit Service d328f3
	} G_STMT_END
Packit Service d328f3
Packit Service d328f3
/*****************************************************************************/
Packit Service d328f3
Packit Service d328f3
struct __nmtst_internal
Packit Service d328f3
{
Packit Service d328f3
	GRand *rand0;
Packit Service d328f3
	guint32 rand_seed;
Packit Service d328f3
	GRand *rand;
Packit Service d328f3
	gboolean is_debug;
Packit Service d328f3
	gboolean assert_logging;
Packit Service d328f3
	gboolean no_expect_message;
Packit Service d328f3
	gboolean test_quick;
Packit Service d328f3
	gboolean test_tap_log;
Packit Service d328f3
	char *sudo_cmd;
Packit Service d328f3
	char **orig_argv;
Packit Service d328f3
};
Packit Service d328f3
Packit Service d328f3
extern struct __nmtst_internal __nmtst_internal;
Packit Service d328f3
Packit Service d328f3
#define NMTST_DEFINE() \
Packit Service d328f3
struct __nmtst_internal __nmtst_internal = { 0 }; \
Packit Service d328f3
\
Packit Service d328f3
__attribute__ ((destructor)) static void \
Packit Service d328f3
_nmtst_exit (void) \
Packit Service d328f3
{ \
Packit Service d328f3
	__nmtst_internal.assert_logging = FALSE; \
Packit Service d328f3
	g_test_assert_expected_messages (); \
Packit Service d328f3
	nmtst_free (); \
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
Packit Service d328f3
static inline gboolean
Packit Service d328f3
nmtst_initialized (void)
Packit Service d328f3
{
Packit Service d328f3
	return !!__nmtst_internal.rand0;
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
#define __NMTST_LOG(cmd, ...) \
Packit Service d328f3
	G_STMT_START { \
Packit Service d328f3
		g_assert (nmtst_initialized ()); \
Packit Service d328f3
		if (!__nmtst_internal.assert_logging || __nmtst_internal.no_expect_message) { \
Packit Service d328f3
			cmd (__VA_ARGS__); \
Packit Service d328f3
		} else { \
Packit Service d328f3
			printf (_NM_UTILS_MACRO_FIRST (__VA_ARGS__) "\n" _NM_UTILS_MACRO_REST (__VA_ARGS__)); \
Packit Service d328f3
		} \
Packit Service d328f3
	} G_STMT_END
Packit Service d328f3
Packit Service d328f3
/* split the string inplace at specific delimiters, allowing escaping with '\\'.
Packit Service d328f3
 * Returns a zero terminated array of pointers into @str.
Packit Service d328f3
 *
Packit Service d328f3
 * The caller must g_free() the returned argv array.
Packit Service d328f3
 **/
Packit Service d328f3
static inline char **
Packit Service d328f3
nmtst_str_split (char *str, const char *delimiters)
Packit Service d328f3
{
Packit Service d328f3
	const char *d;
Packit Service d328f3
	GArray *result = g_array_sized_new (TRUE, FALSE, sizeof (char *), 3);
Packit Service d328f3
Packit Service d328f3
	g_assert (str);
Packit Service d328f3
	g_assert (delimiters && !strchr (delimiters, '\\'));
Packit Service d328f3
Packit Service d328f3
	while (*str) {
Packit Service d328f3
		gsize i = 0, j = 0;
Packit Service d328f3
Packit Service d328f3
		while (TRUE) {
Packit Service d328f3
			char c = str[i];
Packit Service d328f3
Packit Service d328f3
			if (c == '\0') {
Packit Service d328f3
				str[j++] = 0;
Packit Service d328f3
				break;
Packit Service d328f3
			} else if (c == '\\') {
Packit Service d328f3
				str[j++] = str[++i];
Packit Service d328f3
				if (!str[i])
Packit Service d328f3
					break;
Packit Service d328f3
			} else {
Packit Service d328f3
				for (d = delimiters; *d; d++) {
Packit Service d328f3
					if (c == *d) {
Packit Service d328f3
						str[j++] = 0;
Packit Service d328f3
						i++;
Packit Service d328f3
						goto BREAK_INNER_LOOPS;
Packit Service d328f3
					}
Packit Service d328f3
				}
Packit Service d328f3
				str[j++] = c;
Packit Service d328f3
			}
Packit Service d328f3
			i++;
Packit Service d328f3
		}
Packit Service d328f3
Packit Service d328f3
BREAK_INNER_LOOPS:
Packit Service d328f3
		g_array_append_val (result, str);
Packit Service d328f3
		str = &str[i];
Packit Service d328f3
	}
Packit Service d328f3
Packit Service d328f3
	return (char **) g_array_free (result, FALSE);
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
Packit Service d328f3
/* free instances allocated by nmtst (especially nmtst_init()) on shutdown
Packit Service d328f3
 * to release memory. After nmtst_free(), the test is uninitialized again. */
Packit Service d328f3
static inline void
Packit Service d328f3
nmtst_free (void)
Packit Service d328f3
{
Packit Service d328f3
	if (!nmtst_initialized ())
Packit Service d328f3
		return;
Packit Service d328f3
Packit Service d328f3
	g_rand_free (__nmtst_internal.rand0);
Packit Service d328f3
	if (__nmtst_internal.rand)
Packit Service d328f3
		g_rand_free (__nmtst_internal.rand);
Packit Service d328f3
	g_free (__nmtst_internal.sudo_cmd);
Packit Service d328f3
	g_strfreev (__nmtst_internal.orig_argv);
Packit Service d328f3
Packit Service d328f3
	memset (&__nmtst_internal, 0, sizeof (__nmtst_internal));
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static inline void
Packit Service d328f3
_nmtst_log_handler (const gchar   *log_domain,
Packit Service d328f3
                    GLogLevelFlags log_level,
Packit Service d328f3
                    const gchar   *message,
Packit Service d328f3
                    gpointer       user_data)
Packit Service d328f3
{
Packit Service d328f3
	g_print ("%s\n", message);
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static inline void
Packit Service d328f3
__nmtst_init (int *argc, char ***argv, gboolean assert_logging, const char *log_level, const char *log_domains, gboolean *out_set_logging)
Packit Service d328f3
{
Packit Service d328f3
	const char *nmtst_debug;
Packit Service d328f3
	gboolean is_debug = FALSE;
Packit Service d328f3
	char *c_log_level = NULL, *c_log_domains = NULL;
Packit Service d328f3
	char *sudo_cmd = NULL;
Packit Service d328f3
	GArray *debug_messages = g_array_new (TRUE, FALSE, sizeof (char *));
Packit Service d328f3
	int i;
Packit Service d328f3
	gboolean no_expect_message = FALSE;
Packit Service d328f3
	gboolean _out_set_logging;
Packit Service d328f3
	gboolean test_quick = FALSE;
Packit Service d328f3
	gboolean test_quick_set = FALSE;
Packit Service d328f3
	gboolean test_quick_argv = FALSE;
Packit Service d328f3
	gs_unref_ptrarray GPtrArray *p_tests = NULL;
Packit Service d328f3
	gs_unref_ptrarray GPtrArray *s_tests = NULL;
Packit Service d328f3
Packit Service d328f3
	if (!out_set_logging)
Packit Service d328f3
		out_set_logging = &_out_set_logging;
Packit Service d328f3
	*out_set_logging = FALSE;
Packit Service d328f3
Packit Service d328f3
	g_assert (!nmtst_initialized ());
Packit Service d328f3
Packit Service d328f3
	g_assert (!((!!argc) ^ (!!argv)));
Packit Service d328f3
	g_assert (!argc || (g_strv_length (*argv) == *argc));
Packit Service d328f3
	g_assert (!assert_logging || (!log_level && !log_domains));
Packit Service d328f3
Packit Service d328f3
#ifdef __NETWORKMANAGER_UTILS_H__
Packit Service d328f3
	if (!nm_utils_get_testing_initialized ())
Packit Service d328f3
		_nm_utils_set_testing (_NM_UTILS_TEST_GENERAL);
Packit Service d328f3
#endif
Packit Service d328f3
Packit Service d328f3
	if (argc)
Packit Service d328f3
		__nmtst_internal.orig_argv = g_strdupv (*argv);
Packit Service d328f3
Packit Service d328f3
	__nmtst_internal.assert_logging = !!assert_logging;
Packit Service d328f3
Packit Service d328f3
	nm_g_type_init ();
Packit Service d328f3
Packit Service d328f3
	is_debug = g_test_verbose ();
Packit Service d328f3
Packit Service d328f3
	nmtst_debug = g_getenv ("NMTST_DEBUG");
Packit Service d328f3
	if (nmtst_debug) {
Packit Service d328f3
		char **d_argv, **i_argv, *nmtst_debug_copy;
Packit Service d328f3
Packit Service d328f3
		/* By setting then NMTST_DEBUG variable, @is_debug is set automatically.
Packit Service d328f3
		 * This can be reverted with no-debug (on command line or environment variable). */
Packit Service d328f3
		is_debug = TRUE;
Packit Service d328f3
Packit Service d328f3
		nmtst_debug_copy = g_strdup (nmtst_debug);
Packit Service d328f3
		d_argv = nmtst_str_split (nmtst_debug_copy, ",; \t\r\n");
Packit Service d328f3
Packit Service d328f3
		for (i_argv = d_argv; *i_argv; i_argv++) {
Packit Service d328f3
			const char *debug = *i_argv;
Packit Service d328f3
Packit Service d328f3
			if (!g_ascii_strcasecmp (debug, "debug"))
Packit Service d328f3
				is_debug = TRUE;
Packit Service d328f3
			else if (!g_ascii_strcasecmp (debug, "no-debug")) {
Packit Service d328f3
				/* when specifying the NMTST_DEBUG variable, we set is_debug to true. Use this flag to disable this
Packit Service d328f3
				 * (e.g. for only setting the log-level, but not is_debug). */
Packit Service d328f3
				is_debug = FALSE;
Packit Service d328f3
			} else if (!g_ascii_strncasecmp (debug, "log-level=", strlen ("log-level="))) {
Packit Service d328f3
				g_free (c_log_level);
Packit Service d328f3
				log_level = c_log_level = g_strdup (&debug[strlen ("log-level=")]);
Packit Service d328f3
			} else if (!g_ascii_strcasecmp (debug, "D")) {
Packit Service d328f3
				/* shorthand for "log-level=TRACE,no-expect-message" */
Packit Service d328f3
				g_free (c_log_level);
Packit Service d328f3
				log_level = c_log_level = g_strdup ("TRACE");
Packit Service d328f3
				no_expect_message = TRUE;
Packit Service d328f3
			} else if (!g_ascii_strcasecmp (debug, "TRACE")) {
Packit Service d328f3
				g_free (c_log_level);
Packit Service d328f3
				log_level = c_log_level = g_strdup ("TRACE");
Packit Service d328f3
			} else if (!g_ascii_strncasecmp (debug, "log-domains=", strlen ("log-domains="))) {
Packit Service d328f3
				g_free (c_log_domains);
Packit Service d328f3
				log_domains = c_log_domains = g_strdup (&debug[strlen ("log-domains=")]);
Packit Service d328f3
			} else if (!g_ascii_strncasecmp (debug, "sudo-cmd=", strlen ("sudo-cmd="))) {
Packit Service d328f3
				g_free (sudo_cmd);
Packit Service d328f3
				sudo_cmd = g_strdup (&debug[strlen ("sudo-cmd=")]);
Packit Service d328f3
			} else if (!g_ascii_strcasecmp (debug, "no-expect-message")) {
Packit Service d328f3
				no_expect_message = TRUE;
Packit Service d328f3
			} else if (!g_ascii_strncasecmp (debug, "p=", strlen ("p="))) {
Packit Service d328f3
				if (!p_tests)
Packit Service d328f3
					p_tests = g_ptr_array_new_with_free_func (g_free);
Packit Service d328f3
				g_ptr_array_add (p_tests, g_strdup (&debug[strlen ("p=")]));
Packit Service d328f3
			} else if (!g_ascii_strncasecmp (debug, "s=", strlen ("s="))) {
Packit Service d328f3
				if (!s_tests)
Packit Service d328f3
					s_tests = g_ptr_array_new_with_free_func (g_free);
Packit Service d328f3
				g_ptr_array_add (s_tests, g_strdup (&debug[strlen ("s=")]));
Packit Service d328f3
			} else if (!g_ascii_strcasecmp (debug, "slow") || !g_ascii_strcasecmp (debug, "thorough")) {
Packit Service d328f3
				test_quick = FALSE;
Packit Service d328f3
				test_quick_set = TRUE;
Packit Service d328f3
			} else if (!g_ascii_strcasecmp (debug, "quick")) {
Packit Service d328f3
				test_quick = TRUE;
Packit Service d328f3
				test_quick_set = TRUE;
Packit Service d328f3
			} else {
Packit Service d328f3
				char *msg = g_strdup_printf (">>> nmtst: ignore unrecognized NMTST_DEBUG option \"%s\"", debug);
Packit Service d328f3
Packit Service d328f3
				g_array_append_val (debug_messages, msg);
Packit Service d328f3
			}
Packit Service d328f3
		}
Packit Service d328f3
Packit Service d328f3
		g_free (d_argv);
Packit Service d328f3
		g_free (nmtst_debug_copy);
Packit Service d328f3
	}
Packit Service d328f3
Packit Service d328f3
	if (__nmtst_internal.orig_argv) {
Packit Service d328f3
		char **a = __nmtst_internal.orig_argv;
Packit Service d328f3
Packit Service d328f3
		for (; *a; a++) {
Packit Service d328f3
			if (!g_ascii_strcasecmp (*a, "--debug"))
Packit Service d328f3
				is_debug = TRUE;
Packit Service d328f3
			else if (!g_ascii_strcasecmp (*a, "--no-debug"))
Packit Service d328f3
				is_debug = FALSE;
Packit Service d328f3
			else if (   !strcmp (*a, "-m=slow")
Packit Service d328f3
			         || !strcmp (*a, "-m=thorough")
Packit Service d328f3
			         || !strcmp (*a, "-m=quick")
Packit Service d328f3
			         || (!strcmp (*a, "-m") && *(a+1)
Packit Service d328f3
			                                && (   !strcmp (*(a+1), "quick")
Packit Service d328f3
			                                    || !strcmp (*(a+1), "slow")
Packit Service d328f3
			                                    || !strcmp (*(a+1), "thorough"))))
Packit Service d328f3
				test_quick_argv = TRUE;
Packit Service d328f3
			else if (strcmp (*a, "--tap") == 0)
Packit Service d328f3
				__nmtst_internal.test_tap_log = TRUE;
Packit Service d328f3
		}
Packit Service d328f3
	}
Packit Service d328f3
Packit Service d328f3
	if (!argc || g_test_initialized ()) {
Packit Service d328f3
		if (p_tests || s_tests) {
Packit Service d328f3
			char *msg = g_strdup_printf (">>> nmtst: ignore -p and -s options for test which calls g_test_init() itself");
Packit Service d328f3
Packit Service d328f3
			g_array_append_val (debug_messages, msg);
Packit Service d328f3
		}
Packit Service d328f3
	} else {
Packit Service d328f3
		/* We're intentionally assigning a value to static variables
Packit Service d328f3
		 * s_tests_x and p_tests_x without using it afterwards, just
Packit Service d328f3
		 * so that valgrind doesn't complain about the leak. */
Packit Service d328f3
		NM_PRAGMA_WARNING_DISABLE("-Wunused-but-set-variable")
Packit Service d328f3
Packit Service d328f3
		/* g_test_init() is a variadic function, so we cannot pass it
Packit Service d328f3
		 * (variadic) arguments. If you need to pass additional parameters,
Packit Service d328f3
		 * call nmtst_init() with argc==NULL and call g_test_init() yourself. */
Packit Service d328f3
Packit Service d328f3
		/* g_test_init() sets g_log_set_always_fatal() for G_LOG_LEVEL_WARNING
Packit Service d328f3
		 * and G_LOG_LEVEL_CRITICAL. So, beware that the test will fail if you
Packit Service d328f3
		 * have any WARN or ERR log messages -- unless you g_test_expect_message(). */
Packit Service d328f3
		GPtrArray *arg_array = g_ptr_array_new ();
Packit Service d328f3
		gs_free char **arg_array_c = NULL;
Packit Service d328f3
		int arg_array_n, j;
Packit Service d328f3
		static char **s_tests_x, **p_tests_x;
Packit Service d328f3
Packit Service d328f3
		if (*argc) {
Packit Service d328f3
			for (i = 0; i < *argc; i++)
Packit Service d328f3
				g_ptr_array_add (arg_array, (*argv)[i]);
Packit Service d328f3
		} else
Packit Service d328f3
			g_ptr_array_add (arg_array, "./test");
Packit Service d328f3
Packit Service d328f3
		if (test_quick_set && !test_quick_argv)
Packit Service d328f3
			g_ptr_array_add (arg_array, "-m=quick");
Packit Service d328f3
Packit Service d328f3
		if (!__nmtst_internal.test_tap_log) {
Packit Service d328f3
			for (i = 0; p_tests && i < p_tests->len; i++) {
Packit Service d328f3
				g_ptr_array_add (arg_array, "-p");
Packit Service d328f3
				g_ptr_array_add (arg_array, p_tests->pdata[i]);
Packit Service d328f3
			}
Packit Service d328f3
			for (i = 0; s_tests && i < s_tests->len; i++) {
Packit Service d328f3
				g_ptr_array_add (arg_array, "-s");
Packit Service d328f3
				g_ptr_array_add (arg_array, s_tests->pdata[i]);
Packit Service d328f3
			}
Packit Service d328f3
		} else if (p_tests || s_tests) {
Packit Service d328f3
			char *msg = g_strdup_printf (">>> nmtst: ignore -p and -s options for tap-tests");
Packit Service d328f3
Packit Service d328f3
			g_array_append_val (debug_messages, msg);
Packit Service d328f3
		}
Packit Service d328f3
Packit Service d328f3
		g_ptr_array_add (arg_array, NULL);
Packit Service d328f3
Packit Service d328f3
		arg_array_n = arg_array->len - 1;
Packit Service d328f3
		arg_array_c = (char **) g_ptr_array_free (arg_array, FALSE);
Packit Service d328f3
Packit Service d328f3
		g_test_init (&arg_array_n, &arg_array_c, NULL);
Packit Service d328f3
Packit Service d328f3
		if (*argc > 1) {
Packit Service d328f3
			/* collaps argc/argv by removing the arguments detected
Packit Service d328f3
			 * by g_test_init(). */
Packit Service d328f3
			for (i = 1, j = 1; i < *argc; i++) {
Packit Service d328f3
				if ((*argv)[i] == arg_array_c[j])
Packit Service d328f3
					j++;
Packit Service d328f3
				else
Packit Service d328f3
					(*argv)[i] = NULL;
Packit Service d328f3
			}
Packit Service d328f3
			for (i = 1, j = 1; i < *argc; i++) {
Packit Service d328f3
				if ((*argv)[i]) {
Packit Service d328f3
					(*argv)[j++] = (*argv)[i];
Packit Service d328f3
					if (i >= j)
Packit Service d328f3
						(*argv)[i] = NULL;
Packit Service d328f3
				}
Packit Service d328f3
			}
Packit Service d328f3
			*argc = j;
Packit Service d328f3
		}
Packit Service d328f3
Packit Service d328f3
		/* we must "leak" the test paths because they are not cloned by g_test_init(). */
Packit Service d328f3
		if (!__nmtst_internal.test_tap_log) {
Packit Service d328f3
			if (p_tests) {
Packit Service d328f3
				p_tests_x = (char **) g_ptr_array_free (p_tests, FALSE);
Packit Service d328f3
				p_tests = NULL;
Packit Service d328f3
			}
Packit Service d328f3
			if (s_tests) {
Packit Service d328f3
				s_tests_x = (char **) g_ptr_array_free (s_tests, FALSE);
Packit Service d328f3
				s_tests = NULL;
Packit Service d328f3
			}
Packit Service d328f3
		}
Packit Service d328f3
Packit Service d328f3
		NM_PRAGMA_WARNING_REENABLE
Packit Service d328f3
	}
Packit Service d328f3
Packit Service d328f3
	if (test_quick_set)
Packit Service d328f3
		__nmtst_internal.test_quick = test_quick;
Packit Service d328f3
	else if (test_quick_argv)
Packit Service d328f3
		__nmtst_internal.test_quick = g_test_quick ();
Packit Service d328f3
	else {
Packit Service d328f3
#ifdef NMTST_TEST_QUICK
Packit Service d328f3
		__nmtst_internal.test_quick = NMTST_TEST_QUICK;
Packit Service d328f3
#else
Packit Service d328f3
		__nmtst_internal.test_quick = FALSE;
Packit Service d328f3
#endif
Packit Service d328f3
	}
Packit Service d328f3
Packit Service d328f3
	__nmtst_internal.is_debug = is_debug;
Packit Service d328f3
	__nmtst_internal.rand0 = g_rand_new_with_seed (0);
Packit Service d328f3
	__nmtst_internal.sudo_cmd = sudo_cmd;
Packit Service d328f3
	__nmtst_internal.no_expect_message = no_expect_message;
Packit Service d328f3
Packit Service d328f3
	if (!log_level && log_domains) {
Packit Service d328f3
		/* if the log level is not specified (but the domain is), we assume
Packit Service d328f3
		 * the caller wants to set it depending on is_debug */
Packit Service d328f3
		log_level = is_debug ? "DEBUG" : "WARN";
Packit Service d328f3
	}
Packit Service d328f3
Packit Service d328f3
	if (!__nmtst_internal.assert_logging) {
Packit Service d328f3
		gboolean success = TRUE;
Packit Service d328f3
#ifdef _NMTST_INSIDE_CORE
Packit Service d328f3
		success = nm_logging_setup (log_level, log_domains, NULL, NULL);
Packit Service d328f3
		*out_set_logging = TRUE;
Packit Service d328f3
#endif
Packit Service d328f3
		g_assert (success);
Packit Service d328f3
#if GLIB_CHECK_VERSION(2,34,0)
Packit Service d328f3
		if (__nmtst_internal.no_expect_message)
Packit Service d328f3
			g_log_set_always_fatal (G_LOG_FATAL_MASK);
Packit Service d328f3
#else
Packit Service d328f3
		/* g_test_expect_message() is a NOP, so allow any messages */
Packit Service d328f3
		g_log_set_always_fatal (G_LOG_FATAL_MASK);
Packit Service d328f3
#endif
Packit Service d328f3
	} else if (__nmtst_internal.no_expect_message) {
Packit Service d328f3
		/* We have a test that would be assert_logging, but the user specified no_expect_message.
Packit Service d328f3
		 * This transforms g_test_expect_message() into a NOP, but we also have to relax
Packit Service d328f3
		 * g_log_set_always_fatal(), which was set by g_test_init(). */
Packit Service d328f3
		g_log_set_always_fatal (G_LOG_FATAL_MASK);
Packit Service d328f3
#ifdef _NMTST_INSIDE_CORE
Packit Service d328f3
		if (c_log_domains || c_log_level) {
Packit Service d328f3
			/* Normally, tests with assert_logging do not overwrite the logging level/domains because
Packit Service d328f3
			 * the logging statements are part of the assertions. But if the test is run with
Packit Service d328f3
			 * no-expect-message *and* the logging is set explicitly via environment variables,
Packit Service d328f3
			 * we still reset the logging. */
Packit Service d328f3
			gboolean success;
Packit Service d328f3
Packit Service d328f3
			success = nm_logging_setup (log_level, log_domains, NULL, NULL);
Packit Service d328f3
			*out_set_logging = TRUE;
Packit Service d328f3
			g_assert (success);
Packit Service d328f3
		}
Packit Service d328f3
#endif
Packit Service d328f3
	} else {
Packit Service d328f3
#if GLIB_CHECK_VERSION(2,34,0)
Packit Service d328f3
		/* We were called not to set logging levels. This means, that the user
Packit Service d328f3
		 * expects to assert against (all) messages. Any uncought message is fatal. */
Packit Service d328f3
		g_log_set_always_fatal (G_LOG_LEVEL_MASK);
Packit Service d328f3
#else
Packit Service d328f3
		/* g_test_expect_message() is a NOP, so allow any messages */
Packit Service d328f3
		g_log_set_always_fatal (G_LOG_FATAL_MASK);
Packit Service d328f3
#endif
Packit Service d328f3
	}
Packit Service d328f3
Packit Service d328f3
	if ((!__nmtst_internal.assert_logging || (__nmtst_internal.assert_logging && __nmtst_internal.no_expect_message)) &&
Packit Service d328f3
	    (is_debug || (c_log_level && (!g_ascii_strcasecmp (c_log_level, "DEBUG") || !g_ascii_strcasecmp (c_log_level, "TRACE")))) &&
Packit Service d328f3
	    !g_getenv ("G_MESSAGES_DEBUG"))
Packit Service d328f3
	{
Packit Service d328f3
		/* if we are @is_debug or @log_level=="DEBUG" and
Packit Service d328f3
		 * G_MESSAGES_DEBUG is unset, we set G_MESSAGES_DEBUG=all.
Packit Service d328f3
		 * To disable this default behaviour, set G_MESSAGES_DEBUG='' */
Packit Service d328f3
Packit Service d328f3
		/* Note that g_setenv is not thread safe, but you should anyway call
Packit Service d328f3
		 * nmtst_init() at the very start. */
Packit Service d328f3
		g_setenv ("G_MESSAGES_DEBUG", "all", TRUE);
Packit Service d328f3
	}
Packit Service d328f3
Packit Service d328f3
	/* Delay messages until we setup logging. */
Packit Service d328f3
	for (i = 0; i < debug_messages->len; i++)
Packit Service d328f3
		__NMTST_LOG (g_message, "%s", g_array_index (debug_messages, const char *, i));
Packit Service d328f3
Packit Service d328f3
	g_strfreev ((char **) g_array_free (debug_messages, FALSE));
Packit Service d328f3
	g_free (c_log_level);
Packit Service d328f3
	g_free (c_log_domains);
Packit Service d328f3
Packit Service d328f3
#ifdef __NETWORKMANAGER_UTILS_H__
Packit Service d328f3
	/* ensure that monotonic timestamp is called (because it initially logs a line) */
Packit Service d328f3
	nm_utils_get_monotonic_timestamp_s ();
Packit Service d328f3
#endif
Packit Service d328f3
Packit Service d328f3
#ifdef NM_UTILS_H
Packit Service d328f3
	{
Packit Service d328f3
		gs_free_error GError *error = NULL;
Packit Service d328f3
Packit Service d328f3
		if (!nm_utils_init (&error))
Packit Service d328f3
			g_assert_not_reached ();
Packit Service d328f3
		g_assert_no_error (error);
Packit Service d328f3
	}
Packit Service d328f3
#endif
Packit Service d328f3
Packit Service d328f3
	g_log_set_handler (G_LOG_DOMAIN,
Packit Service d328f3
	                   G_LOG_LEVEL_MASK | G_LOG_FLAG_FATAL | G_LOG_FLAG_RECURSION,
Packit Service d328f3
	                   _nmtst_log_handler,
Packit Service d328f3
	                   NULL);
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
#ifndef _NMTST_INSIDE_CORE
Packit Service d328f3
static inline void
Packit Service d328f3
nmtst_init (int *argc, char ***argv, gboolean assert_logging)
Packit Service d328f3
{
Packit Service d328f3
	__nmtst_init (argc, argv, assert_logging, NULL, NULL, NULL);
Packit Service d328f3
}
Packit Service d328f3
#endif
Packit Service d328f3
Packit Service d328f3
static inline gboolean
Packit Service d328f3
nmtst_is_debug (void)
Packit Service d328f3
{
Packit Service d328f3
	g_assert (nmtst_initialized ());
Packit Service d328f3
	return __nmtst_internal.is_debug;
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static inline gboolean
Packit Service d328f3
nmtst_test_quick (void)
Packit Service d328f3
{
Packit Service d328f3
	g_assert (nmtst_initialized ());
Packit Service d328f3
	return __nmtst_internal.test_quick;
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
#if GLIB_CHECK_VERSION(2,34,0)
Packit Service d328f3
#undef g_test_expect_message
Packit Service d328f3
#define g_test_expect_message(...) \
Packit Service d328f3
	G_STMT_START { \
Packit Service d328f3
		g_assert (nmtst_initialized ()); \
Packit Service d328f3
		if (__nmtst_internal.assert_logging && __nmtst_internal.no_expect_message) { \
Packit Service d328f3
			g_debug ("nmtst: assert-logging: g_test_expect_message %s", G_STRINGIFY ((__VA_ARGS__))); \
Packit Service d328f3
		} else { \
Packit Service d328f3
			G_GNUC_BEGIN_IGNORE_DEPRECATIONS \
Packit Service d328f3
			g_test_expect_message (__VA_ARGS__); \
Packit Service d328f3
			G_GNUC_END_IGNORE_DEPRECATIONS \
Packit Service d328f3
		} \
Packit Service d328f3
	} G_STMT_END
Packit Service d328f3
#undef g_test_assert_expected_messages_internal
Packit Service d328f3
#define g_test_assert_expected_messages_internal(domain, file, line, func) \
Packit Service d328f3
	G_STMT_START { \
Packit Service d328f3
		const char *_domain = (domain); \
Packit Service d328f3
		const char *_file = (file); \
Packit Service d328f3
		const char *_func = (func); \
Packit Service d328f3
		int _line = (line); \
Packit Service d328f3
		\
Packit Service d328f3
		if (__nmtst_internal.assert_logging && __nmtst_internal.no_expect_message) \
Packit Service d328f3
			g_debug ("nmtst: assert-logging: g_test_assert_expected_messages(%s, %s:%d, %s)", _domain?:"", _file?:"", _line, _func?:""); \
Packit Service d328f3
		\
Packit Service d328f3
		G_GNUC_BEGIN_IGNORE_DEPRECATIONS \
Packit Service d328f3
		g_test_assert_expected_messages_internal (_domain, _file, _line, _func); \
Packit Service d328f3
		G_GNUC_END_IGNORE_DEPRECATIONS \
Packit Service d328f3
	} G_STMT_END
Packit Service d328f3
#endif
Packit Service d328f3
Packit Service d328f3
/*****************************************************************************/
Packit Service d328f3
Packit Service d328f3
typedef struct _NmtstTestData NmtstTestData;
Packit Service d328f3
Packit Service d328f3
typedef void (*NmtstTestHandler) (const NmtstTestData *test_data);
Packit Service d328f3
Packit Service d328f3
struct _NmtstTestData {
Packit Service d328f3
	union {
Packit Service d328f3
		const char *testpath;
Packit Service d328f3
		char *_testpath;
Packit Service d328f3
	};
Packit Service d328f3
	gsize n_args;
Packit Service d328f3
	gpointer *args;
Packit Service d328f3
	NmtstTestHandler _func_setup;
Packit Service d328f3
	GTestDataFunc _func_test;
Packit Service d328f3
	NmtstTestHandler _func_teardown;
Packit Service d328f3
};
Packit Service d328f3
Packit Service d328f3
static inline void
Packit Service d328f3
_nmtst_test_data_unpack (const NmtstTestData *test_data, gsize n_args, ...)
Packit Service d328f3
{
Packit Service d328f3
	gsize i;
Packit Service d328f3
	va_list ap;
Packit Service d328f3
	gpointer *p;
Packit Service d328f3
Packit Service d328f3
	g_assert (test_data);
Packit Service d328f3
	g_assert_cmpint (n_args, ==, test_data->n_args);
Packit Service d328f3
Packit Service d328f3
	va_start (ap, n_args);
Packit Service d328f3
	for (i = 0; i < n_args; i++) {
Packit Service d328f3
		p = va_arg (ap, gpointer *);
Packit Service d328f3
Packit Service d328f3
		if (p)
Packit Service d328f3
			*p = test_data->args[i];
Packit Service d328f3
	}
Packit Service d328f3
	va_end (ap);
Packit Service d328f3
}
Packit Service d328f3
#define nmtst_test_data_unpack(test_data, ...) _nmtst_test_data_unpack(test_data, NM_NARG (__VA_ARGS__), ##__VA_ARGS__)
Packit Service d328f3
Packit Service d328f3
static inline void
Packit Service d328f3
_nmtst_test_data_free (gpointer data)
Packit Service d328f3
{
Packit Service d328f3
	NmtstTestData *test_data = data;
Packit Service d328f3
Packit Service d328f3
	g_assert (test_data);
Packit Service d328f3
Packit Service d328f3
	g_free (test_data->_testpath);
Packit Service d328f3
	g_free (test_data);
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static inline void
Packit Service d328f3
_nmtst_test_run (gconstpointer data)
Packit Service d328f3
{
Packit Service d328f3
	const NmtstTestData *test_data = data;
Packit Service d328f3
Packit Service d328f3
	if (test_data->_func_setup)
Packit Service d328f3
		test_data->_func_setup (test_data);
Packit Service d328f3
Packit Service d328f3
	test_data->_func_test (test_data);
Packit Service d328f3
Packit Service d328f3
	if (test_data->_func_teardown)
Packit Service d328f3
		test_data->_func_teardown (test_data);
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static inline void
Packit Service d328f3
_nmtst_add_test_func_full (const char *testpath, GTestDataFunc func_test, NmtstTestHandler func_setup, NmtstTestHandler func_teardown, gsize n_args, ...)
Packit Service d328f3
{
Packit Service d328f3
	gsize i;
Packit Service d328f3
	NmtstTestData *data;
Packit Service d328f3
	va_list ap;
Packit Service d328f3
Packit Service d328f3
	g_assert (testpath && testpath[0]);
Packit Service d328f3
	g_assert (func_test);
Packit Service d328f3
Packit Service d328f3
	data = g_malloc0 (sizeof (NmtstTestData) + (sizeof (gpointer) * (n_args + 1)));
Packit Service d328f3
Packit Service d328f3
	data->_testpath = g_strdup (testpath);
Packit Service d328f3
	data->_func_test = func_test;
Packit Service d328f3
	data->_func_setup = func_setup;
Packit Service d328f3
	data->_func_teardown = func_teardown;
Packit Service d328f3
	data->n_args = n_args;
Packit Service d328f3
	data->args = (gpointer) &data[1];
Packit Service d328f3
	va_start (ap, n_args);
Packit Service d328f3
	for (i = 0; i < n_args; i++)
Packit Service d328f3
		data->args[i] = va_arg (ap, gpointer);
Packit Service d328f3
	data->args[i] = NULL;
Packit Service d328f3
	va_end (ap);
Packit Service d328f3
Packit Service d328f3
	g_test_add_data_func_full (testpath,
Packit Service d328f3
	                           data,
Packit Service d328f3
	                           _nmtst_test_run,
Packit Service d328f3
	                           _nmtst_test_data_free);
Packit Service d328f3
}
Packit Service d328f3
#define nmtst_add_test_func_full(testpath, func_test, func_setup, func_teardown, ...) _nmtst_add_test_func_full(testpath, func_test, func_setup, func_teardown, NM_NARG (__VA_ARGS__), ##__VA_ARGS__)
Packit Service d328f3
#define nmtst_add_test_func(testpath, func_test, ...) nmtst_add_test_func_full(testpath, func_test, NULL, NULL, ##__VA_ARGS__)
Packit Service d328f3
Packit Service d328f3
/*****************************************************************************/
Packit Service d328f3
Packit Service d328f3
static inline GRand *
Packit Service d328f3
nmtst_get_rand0 (void)
Packit Service d328f3
{
Packit Service d328f3
	g_assert (nmtst_initialized ());
Packit Service d328f3
	return __nmtst_internal.rand0;
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static inline GRand *
Packit Service d328f3
nmtst_get_rand (void)
Packit Service d328f3
{
Packit Service d328f3
	g_assert (nmtst_initialized ());
Packit Service d328f3
Packit Service d328f3
	if (G_UNLIKELY (!__nmtst_internal.rand)) {
Packit Service d328f3
		guint32 seed;
Packit Service d328f3
		const char *str;
Packit Service d328f3
Packit Service d328f3
		if ((str = g_getenv ("NMTST_SEED_RAND"))) {
Packit Service d328f3
			gchar *s;
Packit Service d328f3
			gint64 i;
Packit Service d328f3
Packit Service d328f3
			i = g_ascii_strtoll (str, &s, 0);
Packit Service d328f3
			g_assert (s[0] == '\0' && i >= 0 && i < G_MAXUINT32);
Packit Service d328f3
Packit Service d328f3
			seed = i;
Packit Service d328f3
			__nmtst_internal.rand = g_rand_new_with_seed (seed);
Packit Service d328f3
		} else {
Packit Service d328f3
			__nmtst_internal.rand = g_rand_new ();
Packit Service d328f3
Packit Service d328f3
			seed = g_rand_int (__nmtst_internal.rand);
Packit Service d328f3
			g_rand_set_seed (__nmtst_internal.rand, seed);
Packit Service d328f3
		}
Packit Service d328f3
		__nmtst_internal.rand_seed = seed;
Packit Service d328f3
Packit Service d328f3
		g_print ("\nnmtst: initialize nmtst_get_rand() with NMTST_SEED_RAND=%u\n", seed);
Packit Service d328f3
	}
Packit Service d328f3
	return __nmtst_internal.rand;
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static inline guint32
Packit Service d328f3
nmtst_get_rand_int (void)
Packit Service d328f3
{
Packit Service d328f3
	return g_rand_int (nmtst_get_rand ());
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static inline gpointer
Packit Service d328f3
nmtst_rand_buf (GRand *rand, gpointer buffer, gsize buffer_length)
Packit Service d328f3
{
Packit Service d328f3
	guint32 v;
Packit Service d328f3
	guint8 *b = buffer;
Packit Service d328f3
Packit Service d328f3
	if (!buffer_length)
Packit Service d328f3
		return buffer;
Packit Service d328f3
Packit Service d328f3
	g_assert (buffer);
Packit Service d328f3
Packit Service d328f3
	if (!rand)
Packit Service d328f3
		rand = nmtst_get_rand ();
Packit Service d328f3
Packit Service d328f3
	for (; buffer_length >= sizeof (guint32); buffer_length -= sizeof (guint32), b += sizeof (guint32)) {
Packit Service d328f3
		v = g_rand_int (rand);
Packit Service d328f3
		memcpy (b, &v, sizeof (guint32));
Packit Service d328f3
	}
Packit Service d328f3
	if (buffer_length > 0) {
Packit Service d328f3
		v = g_rand_int (rand);
Packit Service d328f3
		do {
Packit Service d328f3
			*(b++) = v & 0xFF;
Packit Service d328f3
			v >>= 8;
Packit Service d328f3
		} while (--buffer_length > 0);
Packit Service d328f3
	}
Packit Service d328f3
	return buffer;
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static inline void *
Packit Service d328f3
nmtst_rand_perm (GRand *rand, void *dst, const void *src, gsize elmt_size, gsize n_elmt)
Packit Service d328f3
{
Packit Service d328f3
	gsize i, j;
Packit Service d328f3
	char *p_, *pj;
Packit Service d328f3
	char *bu;
Packit Service d328f3
Packit Service d328f3
	g_assert (dst);
Packit Service d328f3
	g_assert (elmt_size > 0);
Packit Service d328f3
	g_assert (n_elmt < G_MAXINT32);
Packit Service d328f3
Packit Service d328f3
	if (n_elmt == 0)
Packit Service d328f3
		return dst;
Packit Service d328f3
Packit Service d328f3
	if (src && dst != src)
Packit Service d328f3
		memcpy (dst, src, elmt_size * n_elmt);
Packit Service d328f3
Packit Service d328f3
	if (!rand)
Packit Service d328f3
		rand = nmtst_get_rand ();
Packit Service d328f3
Packit Service d328f3
	bu = g_slice_alloc (elmt_size);
Packit Service d328f3
Packit Service d328f3
	p_ = dst;
Packit Service d328f3
	for (i = n_elmt; i > 1; i--) {
Packit Service d328f3
		j = g_rand_int_range (rand, 0, i);
Packit Service d328f3
Packit Service d328f3
		if (j != 0) {
Packit Service d328f3
			pj = &p_[j * elmt_size];
Packit Service d328f3
Packit Service d328f3
			/* swap */
Packit Service d328f3
			memcpy (bu, p_, elmt_size);
Packit Service d328f3
			memcpy (p_, pj, elmt_size);
Packit Service d328f3
			memcpy (pj, bu, elmt_size);
Packit Service d328f3
		}
Packit Service d328f3
		p_ += elmt_size;
Packit Service d328f3
	}
Packit Service d328f3
Packit Service d328f3
	g_slice_free1 (elmt_size, bu);
Packit Service d328f3
	return dst;
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static inline GSList *
Packit Service d328f3
nmtst_rand_perm_gslist (GRand *rand, GSList *list)
Packit Service d328f3
{
Packit Service d328f3
	GSList *result;
Packit Service d328f3
	guint l;
Packit Service d328f3
Packit Service d328f3
	if (!rand)
Packit Service d328f3
		rand = nmtst_get_rand ();
Packit Service d328f3
Packit Service d328f3
	/* no need for an efficient implementation :) */
Packit Service d328f3
Packit Service d328f3
	result = 0;
Packit Service d328f3
	for (l = g_slist_length (list); l > 0; l--) {
Packit Service d328f3
		GSList *tmp;
Packit Service d328f3
Packit Service d328f3
		tmp = g_slist_nth (list, g_rand_int (rand) % l);
Packit Service d328f3
		g_assert (tmp);
Packit Service d328f3
Packit Service d328f3
		list = g_slist_remove_link (list, tmp);
Packit Service d328f3
		result = g_slist_concat (tmp, result);
Packit Service d328f3
	}
Packit Service d328f3
	g_assert (!list);
Packit Service d328f3
	return result;
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
/*****************************************************************************/
Packit Service d328f3
Packit Service d328f3
static inline gboolean
Packit Service d328f3
_nmtst_main_loop_run_timeout (gpointer user_data)
Packit Service d328f3
{
Packit Service d328f3
	GMainLoop **p_loop = user_data;
Packit Service d328f3
Packit Service d328f3
	g_assert (p_loop);
Packit Service d328f3
	g_assert (*p_loop);
Packit Service d328f3
Packit Service d328f3
	g_main_loop_quit (*p_loop);
Packit Service d328f3
	*p_loop = NULL;
Packit Service d328f3
Packit Service d328f3
	return G_SOURCE_REMOVE;
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static inline gboolean
Packit Service d328f3
nmtst_main_loop_run (GMainLoop *loop, int timeout_ms)
Packit Service d328f3
{
Packit Service d328f3
	GSource *source = NULL;
Packit Service d328f3
	guint id = 0;
Packit Service d328f3
	GMainLoop *loopx = loop;
Packit Service d328f3
Packit Service d328f3
	if (timeout_ms > 0) {
Packit Service d328f3
		source = g_timeout_source_new (timeout_ms);
Packit Service d328f3
		g_source_set_callback (source, _nmtst_main_loop_run_timeout, &loopx, NULL);
Packit Service d328f3
		id = g_source_attach (source, g_main_loop_get_context (loop));
Packit Service d328f3
		g_assert (id);
Packit Service d328f3
		g_source_unref (source);
Packit Service d328f3
	}
Packit Service d328f3
Packit Service d328f3
	g_main_loop_run (loop);
Packit Service d328f3
Packit Service d328f3
	/* if the timeout was reached, return FALSE. */
Packit Service d328f3
	return loopx != NULL;
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static inline void
Packit Service d328f3
_nmtst_main_loop_quit_on_notify (GObject *object, GParamSpec *pspec, gpointer user_data)
Packit Service d328f3
{
Packit Service d328f3
	GMainLoop *loop = user_data;
Packit Service d328f3
Packit Service d328f3
	g_assert (G_IS_OBJECT (object));
Packit Service d328f3
	g_assert (loop);
Packit Service d328f3
Packit Service d328f3
	g_main_loop_quit (loop);
Packit Service d328f3
}
Packit Service d328f3
#define nmtst_main_loop_quit_on_notify ((GCallback) _nmtst_main_loop_quit_on_notify)
Packit Service d328f3
Packit Service d328f3
/*****************************************************************************/
Packit Service d328f3
Packit Service d328f3
static inline const char *
Packit Service d328f3
nmtst_get_sudo_cmd (void)
Packit Service d328f3
{
Packit Service d328f3
	g_assert (nmtst_initialized ());
Packit Service d328f3
	return __nmtst_internal.sudo_cmd;
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static inline void
Packit Service d328f3
nmtst_reexec_sudo (void)
Packit Service d328f3
{
Packit Service d328f3
	char *str;
Packit Service d328f3
	char **argv;
Packit Service d328f3
	int i;
Packit Service d328f3
	int errsv;
Packit Service d328f3
Packit Service d328f3
	g_assert (nmtst_initialized ());
Packit Service d328f3
	g_assert (__nmtst_internal.orig_argv);
Packit Service d328f3
Packit Service d328f3
	if (!__nmtst_internal.sudo_cmd)
Packit Service d328f3
		return;
Packit Service d328f3
Packit Service d328f3
	str = g_strjoinv (" ", __nmtst_internal.orig_argv);
Packit Service d328f3
	__NMTST_LOG (g_message, ">> exec %s %s", __nmtst_internal.sudo_cmd, str);
Packit Service d328f3
Packit Service d328f3
	argv = g_new0 (char *, 1 + g_strv_length (__nmtst_internal.orig_argv) + 1);
Packit Service d328f3
	argv[0] = __nmtst_internal.sudo_cmd;
Packit Service d328f3
	for (i = 0; __nmtst_internal.orig_argv[i]; i++)
Packit Service d328f3
		argv[i+1] = __nmtst_internal.orig_argv[i];
Packit Service d328f3
Packit Service d328f3
	execvp (__nmtst_internal.sudo_cmd, argv);
Packit Service d328f3
Packit Service d328f3
	errsv = errno;
Packit Service d328f3
	g_error (">> exec %s failed: %d - %s", __nmtst_internal.sudo_cmd, errsv, strerror (errsv));
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
/*****************************************************************************/
Packit Service d328f3
Packit Service d328f3
static inline gsize
Packit Service d328f3
nmtst_find_all_indexes (gpointer *elements,
Packit Service d328f3
                        gsize n_elements,
Packit Service d328f3
                        gpointer *needles,
Packit Service d328f3
                        gsize n_needles,
Packit Service d328f3
                        gboolean (*equal_fcn) (gpointer element, gpointer needle, gpointer user_data),
Packit Service d328f3
                        gpointer user_data,
Packit Service d328f3
                        gssize *out_idx)
Packit Service d328f3
{
Packit Service d328f3
	gsize i, j, k;
Packit Service d328f3
	gsize found = 0;
Packit Service d328f3
Packit Service d328f3
	for (i = 0; i < n_needles; i++) {
Packit Service d328f3
		gssize idx = -1;
Packit Service d328f3
Packit Service d328f3
		for (j = 0; j < n_elements; j++) {
Packit Service d328f3
Packit Service d328f3
			/* no duplicates */
Packit Service d328f3
			for (k = 0; k < i; k++) {
Packit Service d328f3
				if (out_idx[k] == j)
Packit Service d328f3
					goto next;
Packit Service d328f3
			}
Packit Service d328f3
Packit Service d328f3
			if (equal_fcn (elements[j], needles[i], user_data)) {
Packit Service d328f3
				idx = j;
Packit Service d328f3
				break;
Packit Service d328f3
			}
Packit Service d328f3
next:
Packit Service d328f3
			;
Packit Service d328f3
		}
Packit Service d328f3
Packit Service d328f3
		out_idx[i] = idx;
Packit Service d328f3
		if (idx >= 0)
Packit Service d328f3
			found++;
Packit Service d328f3
	}
Packit Service d328f3
Packit Service d328f3
	return found;
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
/*****************************************************************************/
Packit Service d328f3
Packit Service d328f3
#define __define_nmtst_static(NUM,SIZE) \
Packit Service d328f3
static inline const char * \
Packit Service d328f3
nmtst_static_##SIZE##_##NUM (const char *str) \
Packit Service d328f3
{ \
Packit Service d328f3
	gsize l; \
Packit Service d328f3
	static char buf[SIZE]; \
Packit Service d328f3
\
Packit Service d328f3
	if (!str) \
Packit Service d328f3
		return NULL; \
Packit Service d328f3
	l = g_strlcpy (buf, str, sizeof (buf)); \
Packit Service d328f3
	g_assert (l < sizeof (buf)); \
Packit Service d328f3
	return buf; \
Packit Service d328f3
}
Packit Service d328f3
__define_nmtst_static(01, 1024)
Packit Service d328f3
__define_nmtst_static(02, 1024)
Packit Service d328f3
__define_nmtst_static(03, 1024)
Packit Service d328f3
#undef __define_nmtst_static
Packit Service d328f3
Packit Service d328f3
#define NMTST_UUID_INIT(uuid) \
Packit Service d328f3
	gs_free char *_nmtst_hidden_##uuid = nm_utils_uuid_generate (); \
Packit Service d328f3
	const char *const uuid = _nmtst_hidden_##uuid
Packit Service d328f3
Packit Service d328f3
static inline const char *
Packit Service d328f3
nmtst_uuid_generate (void)
Packit Service d328f3
{
Packit Service d328f3
	static char u[37];
Packit Service d328f3
	gs_free char *m = NULL;
Packit Service d328f3
Packit Service d328f3
	m = nm_utils_uuid_generate ();
Packit Service d328f3
	g_assert (m && strlen (m) == sizeof (u) - 1);
Packit Service d328f3
	memcpy (u, m, sizeof (u));
Packit Service d328f3
	return u;
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
#define NMTST_SWAP(x,y) \
Packit Service d328f3
	G_STMT_START { \
Packit Service d328f3
		char __nmtst_swap_temp[sizeof(x) == sizeof(y) ? (signed) sizeof(x) : -1]; \
Packit Service d328f3
		memcpy(__nmtst_swap_temp, &y, sizeof(x)); \
Packit Service d328f3
		memcpy(&y,                &x, sizeof(x)); \
Packit Service d328f3
		memcpy(&x, __nmtst_swap_temp, sizeof(x)); \
Packit Service d328f3
	} G_STMT_END
Packit Service d328f3
Packit Service d328f3
#define nmtst_assert_str_has_substr(str, substr) \
Packit Service d328f3
	G_STMT_START { \
Packit Service d328f3
		const char *__str = (str); \
Packit Service d328f3
		const char *__substr = (substr); \
Packit Service d328f3
		\
Packit Service d328f3
		g_assert (__str); \
Packit Service d328f3
		g_assert (__substr); \
Packit Service d328f3
		if (strstr (__str, __substr) == NULL) \
Packit Service d328f3
			g_error ("%s:%d: Expects \"%s\" but got \"%s\"", __FILE__, __LINE__, __substr, __str); \
Packit Service d328f3
	} G_STMT_END
Packit Service d328f3
Packit Service d328f3
static inline guint32
Packit Service d328f3
nmtst_inet4_from_string (const char *str)
Packit Service d328f3
{
Packit Service d328f3
	guint32 addr;
Packit Service d328f3
	int success;
Packit Service d328f3
Packit Service d328f3
	if (!str)
Packit Service d328f3
		return 0;
Packit Service d328f3
Packit Service d328f3
	success = inet_pton (AF_INET, str, &addr);
Packit Service d328f3
Packit Service d328f3
	g_assert (success == 1);
Packit Service d328f3
Packit Service d328f3
	return addr;
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static inline const struct in6_addr *
Packit Service d328f3
nmtst_inet6_from_string (const char *str)
Packit Service d328f3
{
Packit Service d328f3
	static struct in6_addr addr;
Packit Service d328f3
	int success;
Packit Service d328f3
Packit Service d328f3
	if (!str)
Packit Service d328f3
		addr = in6addr_any;
Packit Service d328f3
	else {
Packit Service d328f3
		success = inet_pton (AF_INET6, str, &addr);
Packit Service d328f3
		g_assert (success == 1);
Packit Service d328f3
	}
Packit Service d328f3
Packit Service d328f3
	return &addr;
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static inline void
Packit Service d328f3
_nmtst_assert_ip4_address (const char *file, int line, in_addr_t addr, const char *str_expected)
Packit Service d328f3
{
Packit Service d328f3
	if (nmtst_inet4_from_string (str_expected) != addr) {
Packit Service d328f3
		char buf[100];
Packit Service d328f3
Packit Service d328f3
		g_error ("%s:%d: Unexpected IPv4 address: expected %s, got %s",
Packit Service d328f3
		         file, line, str_expected ? str_expected : "0.0.0.0",
Packit Service d328f3
		         inet_ntop (AF_INET, &addr, buf, sizeof (buf)));
Packit Service d328f3
	}
Packit Service d328f3
}
Packit Service d328f3
#define nmtst_assert_ip4_address(addr, str_expected) _nmtst_assert_ip4_address (__FILE__, __LINE__, addr, str_expected)
Packit Service d328f3
Packit Service d328f3
static inline void
Packit Service d328f3
_nmtst_assert_ip6_address (const char *file, int line, const struct in6_addr *addr, const char *str_expected)
Packit Service d328f3
{
Packit Service d328f3
	struct in6_addr any = in6addr_any;
Packit Service d328f3
Packit Service d328f3
	if (!addr)
Packit Service d328f3
		addr = &any;
Packit Service d328f3
Packit Service d328f3
	if (memcmp (nmtst_inet6_from_string (str_expected), addr, sizeof (*addr)) != 0) {
Packit Service d328f3
		char buf[100];
Packit Service d328f3
Packit Service d328f3
		g_error ("%s:%d: Unexpected IPv6 address: expected %s, got %s",
Packit Service d328f3
		         file, line, str_expected ? str_expected : "::",
Packit Service d328f3
		         inet_ntop (AF_INET6, addr, buf, sizeof (buf)));
Packit Service d328f3
	}
Packit Service d328f3
}
Packit Service d328f3
#define nmtst_assert_ip6_address(addr, str_expected) _nmtst_assert_ip6_address (__FILE__, __LINE__, addr, str_expected)
Packit Service d328f3
Packit Service d328f3
#define nmtst_spawn_sync(working_directory, standard_out, standard_err, assert_exit_status, ...) \
Packit Service d328f3
	__nmtst_spawn_sync (working_directory, standard_out, standard_err, assert_exit_status, ##__VA_ARGS__, NULL)
Packit Service d328f3
static inline gint
Packit Service d328f3
__nmtst_spawn_sync (const char *working_directory, char **standard_out, char **standard_err, int assert_exit_status, ...) G_GNUC_NULL_TERMINATED;
Packit Service d328f3
static inline gint
Packit Service d328f3
__nmtst_spawn_sync (const char *working_directory, char **standard_out, char **standard_err, int assert_exit_status, ...)
Packit Service d328f3
{
Packit Service d328f3
	gint exit_status = 0;
Packit Service d328f3
	GError *error = NULL;
Packit Service d328f3
	char *arg;
Packit Service d328f3
	va_list va_args;
Packit Service d328f3
	GPtrArray *argv = g_ptr_array_new ();
Packit Service d328f3
	gboolean success;
Packit Service d328f3
Packit Service d328f3
	va_start (va_args, assert_exit_status);
Packit Service d328f3
	while ((arg = va_arg (va_args, char *)))
Packit Service d328f3
		g_ptr_array_add (argv, arg);
Packit Service d328f3
	va_end (va_args);
Packit Service d328f3
Packit Service d328f3
	g_assert (argv->len >= 1);
Packit Service d328f3
	g_ptr_array_add (argv, NULL);
Packit Service d328f3
Packit Service d328f3
	success = g_spawn_sync (working_directory,
Packit Service d328f3
	                        (char**) argv->pdata,
Packit Service d328f3
	                        NULL,
Packit Service d328f3
	                        0 /*G_SPAWN_DEFAULT*/,
Packit Service d328f3
	                        NULL,
Packit Service d328f3
	                        NULL,
Packit Service d328f3
	                        standard_out,
Packit Service d328f3
	                        standard_err,
Packit Service d328f3
	                        &exit_status,
Packit Service d328f3
	                        &error);
Packit Service d328f3
	if (!success)
Packit Service d328f3
		g_error ("nmtst_spawn_sync(%s): %s", ((char **) argv->pdata)[0], error->message);
Packit Service d328f3
	g_assert (!error);
Packit Service d328f3
Packit Service d328f3
	g_assert (!standard_out || *standard_out);
Packit Service d328f3
	g_assert (!standard_err || *standard_err);
Packit Service d328f3
Packit Service d328f3
	if (assert_exit_status != -1) {
Packit Service d328f3
		/* exit status is a guint8 on success. Set @assert_exit_status to -1
Packit Service d328f3
		 * not to check for the exit status. */
Packit Service d328f3
		g_assert (WIFEXITED (exit_status));
Packit Service d328f3
		g_assert_cmpint (WEXITSTATUS (exit_status), ==, assert_exit_status);
Packit Service d328f3
	}
Packit Service d328f3
Packit Service d328f3
	g_ptr_array_free (argv, TRUE);
Packit Service d328f3
	return exit_status;
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
/*****************************************************************************/
Packit Service d328f3
Packit Service d328f3
static inline char *
Packit Service d328f3
nmtst_file_resolve_relative_path (const char *rel, const char *cwd)
Packit Service d328f3
{
Packit Service d328f3
	gs_free char *cwd_free = NULL;
Packit Service d328f3
Packit Service d328f3
	g_assert (rel && *rel);
Packit Service d328f3
Packit Service d328f3
	if (g_path_is_absolute (rel))
Packit Service d328f3
		return g_strdup (rel);
Packit Service d328f3
Packit Service d328f3
	if (!cwd)
Packit Service d328f3
		cwd = cwd_free = g_get_current_dir ();
Packit Service d328f3
	return g_build_filename (cwd, rel, NULL);
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static inline char *
Packit Service d328f3
nmtst_file_get_contents (const char *filename)
Packit Service d328f3
{
Packit Service d328f3
	GError *error = NULL;
Packit Service d328f3
	gboolean success;
Packit Service d328f3
	char *contents = NULL;
Packit Service d328f3
	gsize len;
Packit Service d328f3
Packit Service d328f3
	success = g_file_get_contents (filename, &contents, &len, &error);
Packit Service d328f3
	nmtst_assert_success (success && contents, error);
Packit Service d328f3
	g_assert_cmpint (strlen (contents), ==, len);
Packit Service d328f3
	return contents;
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
#define nmtst_file_set_contents(filename, content) \
Packit Service d328f3
	G_STMT_START { \
Packit Service d328f3
		GError *_error = NULL; \
Packit Service d328f3
		gboolean _success; \
Packit Service d328f3
		\
Packit Service d328f3
		_success = g_file_set_contents ((filename), (content), -1, &_error); \
Packit Service d328f3
		nmtst_assert_success (_success, _error); \
Packit Service d328f3
	} G_STMT_END
Packit Service d328f3
Packit Service d328f3
/*****************************************************************************/
Packit Service d328f3
Packit Service d328f3
static inline void
Packit Service d328f3
nmtst_file_unlink_if_exists (const char *name)
Packit Service d328f3
{
Packit Service d328f3
	int errsv;
Packit Service d328f3
Packit Service d328f3
	g_assert (name && name[0]);
Packit Service d328f3
Packit Service d328f3
	if (unlink (name) != 0) {
Packit Service d328f3
		errsv = errno;
Packit Service d328f3
		if (errsv != ENOENT)
Packit Service d328f3
			g_error ("nmtst_file_unlink_if_exists(%s): failed with %s", name, strerror (errsv));
Packit Service d328f3
	}
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static inline void
Packit Service d328f3
nmtst_file_unlink (const char *name)
Packit Service d328f3
{
Packit Service d328f3
	int errsv;
Packit Service d328f3
Packit Service d328f3
	g_assert (name && name[0]);
Packit Service d328f3
Packit Service d328f3
	if (unlink (name) != 0) {
Packit Service d328f3
		errsv = errno;
Packit Service d328f3
		g_error ("nmtst_file_unlink(%s): failed with %s", name, strerror (errsv));
Packit Service d328f3
	}
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static inline void
Packit Service d328f3
_nmtst_auto_unlinkfile (char **p_name)
Packit Service d328f3
{
Packit Service d328f3
	if (*p_name) {
Packit Service d328f3
		nmtst_file_unlink (*p_name);
Packit Service d328f3
		nm_clear_g_free (p_name);
Packit Service d328f3
	}
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
#define nmtst_auto_unlinkfile nm_auto(_nmtst_auto_unlinkfile)
Packit Service d328f3
Packit Service d328f3
/*****************************************************************************/
Packit Service d328f3
Packit Service d328f3
static inline void
Packit Service d328f3
_nmtst_assert_resolve_relative_path_equals (const char *f1, const char *f2, const char *file, int line)
Packit Service d328f3
{
Packit Service d328f3
	gs_free char *p1 = NULL, *p2 = NULL;
Packit Service d328f3
Packit Service d328f3
	p1 = nmtst_file_resolve_relative_path (f1, NULL);
Packit Service d328f3
	p2 = nmtst_file_resolve_relative_path (f2, NULL);
Packit Service d328f3
	g_assert (p1 && *p1);
Packit Service d328f3
Packit Service d328f3
	/* Fixme: later we might need to coalesce repeated '/', "./", and "../".
Packit Service d328f3
	 * For now, it's good enough. */
Packit Service d328f3
	if (g_strcmp0 (p1, p2) != 0)
Packit Service d328f3
		g_error ("%s:%d : filenames don't match \"%s\" vs. \"%s\" // \"%s\" - \"%s\"", file, line, f1, f2, p1, p2);
Packit Service d328f3
}
Packit Service d328f3
#define nmtst_assert_resolve_relative_path_equals(f1, f2) _nmtst_assert_resolve_relative_path_equals (f1, f2, __FILE__, __LINE__);
Packit Service d328f3
Packit Service d328f3
/*****************************************************************************/
Packit Service d328f3
Packit Service d328f3
#ifdef NM_SETTING_IP_CONFIG_H
Packit Service d328f3
static inline void
Packit Service d328f3
nmtst_setting_ip_config_add_address (NMSettingIPConfig *s_ip,
Packit Service d328f3
                                     const char *address,
Packit Service d328f3
                                     guint prefix)
Packit Service d328f3
{
Packit Service d328f3
	NMIPAddress *addr;
Packit Service d328f3
	int family;
Packit Service d328f3
Packit Service d328f3
	g_assert (s_ip);
Packit Service d328f3
Packit Service d328f3
	if (nm_utils_ipaddr_valid (AF_INET, address))
Packit Service d328f3
		family = AF_INET;
Packit Service d328f3
	else if (nm_utils_ipaddr_valid (AF_INET6, address))
Packit Service d328f3
		family = AF_INET6;
Packit Service d328f3
	else
Packit Service d328f3
		g_assert_not_reached ();
Packit Service d328f3
Packit Service d328f3
	addr = nm_ip_address_new (family, address, prefix, NULL);
Packit Service d328f3
	g_assert (addr);
Packit Service d328f3
	g_assert (nm_setting_ip_config_add_address (s_ip, addr));
Packit Service d328f3
	nm_ip_address_unref (addr);
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static inline void
Packit Service d328f3
nmtst_setting_ip_config_add_route (NMSettingIPConfig *s_ip,
Packit Service d328f3
                                   const char *dest,
Packit Service d328f3
                                   guint prefix,
Packit Service d328f3
                                   const char *next_hop,
Packit Service d328f3
                                   gint64 metric)
Packit Service d328f3
{
Packit Service d328f3
	NMIPRoute *route;
Packit Service d328f3
	int family;
Packit Service d328f3
Packit Service d328f3
	g_assert (s_ip);
Packit Service d328f3
Packit Service d328f3
	if (nm_utils_ipaddr_valid (AF_INET, dest))
Packit Service d328f3
		family = AF_INET;
Packit Service d328f3
	else if (nm_utils_ipaddr_valid (AF_INET6, dest))
Packit Service d328f3
		family = AF_INET6;
Packit Service d328f3
	else
Packit Service d328f3
		g_assert_not_reached ();
Packit Service d328f3
Packit Service d328f3
	route = nm_ip_route_new (family, dest, prefix, next_hop, metric, NULL);
Packit Service d328f3
	g_assert (route);
Packit Service d328f3
	g_assert (nm_setting_ip_config_add_route (s_ip, route));
Packit Service d328f3
	nm_ip_route_unref (route);
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static inline void
Packit Service d328f3
nmtst_assert_route_attribute_string (NMIPRoute *route, const char *name, const char *value)
Packit Service d328f3
{
Packit Service d328f3
	GVariant *variant;
Packit Service d328f3
Packit Service d328f3
	variant = nm_ip_route_get_attribute (route, name);
Packit Service d328f3
	g_assert (variant);
Packit Service d328f3
	g_assert (g_variant_is_of_type (variant, G_VARIANT_TYPE_STRING));
Packit Service d328f3
	g_assert_cmpstr (g_variant_get_string (variant, NULL), ==, value);
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static inline void
Packit Service d328f3
nmtst_assert_route_attribute_byte (NMIPRoute *route, const char *name, guchar value)
Packit Service d328f3
{
Packit Service d328f3
	GVariant *variant;
Packit Service d328f3
Packit Service d328f3
	variant = nm_ip_route_get_attribute (route, name);
Packit Service d328f3
	g_assert (variant);
Packit Service d328f3
	g_assert (g_variant_is_of_type (variant, G_VARIANT_TYPE_BYTE));
Packit Service d328f3
	g_assert_cmpint (g_variant_get_byte (variant), ==, value);
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static inline void
Packit Service d328f3
nmtst_assert_route_attribute_uint32 (NMIPRoute *route, const char *name, guint32 value)
Packit Service d328f3
{
Packit Service d328f3
	GVariant *variant;
Packit Service d328f3
Packit Service d328f3
	variant = nm_ip_route_get_attribute (route, name);
Packit Service d328f3
	g_assert (variant);
Packit Service d328f3
	g_assert (g_variant_is_of_type (variant, G_VARIANT_TYPE_UINT32));
Packit Service d328f3
	g_assert_cmpint (g_variant_get_uint32 (variant), ==, value);
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static inline void
Packit Service d328f3
nmtst_assert_route_attribute_boolean (NMIPRoute *route, const char *name, gboolean value)
Packit Service d328f3
{
Packit Service d328f3
	GVariant *variant;
Packit Service d328f3
Packit Service d328f3
	variant = nm_ip_route_get_attribute (route, name);
Packit Service d328f3
	g_assert (variant);
Packit Service d328f3
	g_assert (g_variant_is_of_type (variant, G_VARIANT_TYPE_BOOLEAN));
Packit Service d328f3
	g_assert_cmpint (g_variant_get_boolean (variant), ==, value);
Packit Service d328f3
}
Packit Service d328f3
#endif /* NM_SETTING_IP_CONFIG_H */
Packit Service d328f3
Packit Service d328f3
#if (defined(__NM_SIMPLE_CONNECTION_H__) && defined(__NM_SETTING_CONNECTION_H__)) || (defined(NM_CONNECTION_H))
Packit Service d328f3
Packit Service d328f3
static inline NMConnection *
Packit Service d328f3
nmtst_clone_connection (NMConnection *connection)
Packit Service d328f3
{
Packit Service d328f3
	g_assert (NM_IS_CONNECTION (connection));
Packit Service d328f3
Packit Service d328f3
#if defined(__NM_SIMPLE_CONNECTION_H__)
Packit Service d328f3
	return nm_simple_connection_new_clone (connection);
Packit Service d328f3
#else
Packit Service d328f3
	return nm_connection_duplicate (connection);
Packit Service d328f3
#endif
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static inline NMConnection *
Packit Service d328f3
nmtst_create_minimal_connection (const char *id, const char *uuid, const char *type, NMSettingConnection **out_s_con)
Packit Service d328f3
{
Packit Service d328f3
	NMConnection *con;
Packit Service d328f3
	NMSetting *s_base = NULL;
Packit Service d328f3
	NMSettingConnection *s_con;
Packit Service d328f3
	gs_free char *uuid_free = NULL;
Packit Service d328f3
Packit Service d328f3
	g_assert (id);
Packit Service d328f3
Packit Service d328f3
	if (uuid)
Packit Service d328f3
		g_assert (nm_utils_is_uuid (uuid));
Packit Service d328f3
	else
Packit Service d328f3
		uuid = uuid_free = nm_utils_uuid_generate ();
Packit Service d328f3
Packit Service d328f3
	if (type) {
Packit Service d328f3
		GType type_g;
Packit Service d328f3
Packit Service d328f3
#if defined(__NM_SIMPLE_CONNECTION_H__)
Packit Service d328f3
		type_g = nm_setting_lookup_type (type);
Packit Service d328f3
#else
Packit Service d328f3
		type_g = nm_connection_lookup_setting_type (type);
Packit Service d328f3
#endif
Packit Service d328f3
Packit Service d328f3
		g_assert (type_g != G_TYPE_INVALID);
Packit Service d328f3
Packit Service d328f3
		s_base = g_object_new (type_g, NULL);
Packit Service d328f3
		g_assert (NM_IS_SETTING (s_base));
Packit Service d328f3
	}
Packit Service d328f3
Packit Service d328f3
#if defined(__NM_SIMPLE_CONNECTION_H__)
Packit Service d328f3
	con = nm_simple_connection_new ();
Packit Service d328f3
#else
Packit Service d328f3
	con = nm_connection_new ();
Packit Service d328f3
#endif
Packit Service d328f3
Packit Service d328f3
	s_con = NM_SETTING_CONNECTION (nm_setting_connection_new ());
Packit Service d328f3
Packit Service d328f3
	g_object_set (s_con,
Packit Service d328f3
	              NM_SETTING_CONNECTION_ID, id,
Packit Service d328f3
	              NM_SETTING_CONNECTION_UUID, uuid,
Packit Service d328f3
	              NM_SETTING_CONNECTION_TYPE, type,
Packit Service d328f3
	              NULL);
Packit Service d328f3
	nm_connection_add_setting (con, NM_SETTING (s_con));
Packit Service d328f3
Packit Service d328f3
	if (s_base)
Packit Service d328f3
		nm_connection_add_setting (con, s_base);
Packit Service d328f3
Packit Service d328f3
	if (out_s_con)
Packit Service d328f3
		*out_s_con = s_con;
Packit Service d328f3
	return con;
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static inline gboolean
Packit Service d328f3
_nmtst_connection_normalize_v (NMConnection *connection, va_list args)
Packit Service d328f3
{
Packit Service d328f3
	GError *error = NULL;
Packit Service d328f3
	gboolean success;
Packit Service d328f3
	gboolean was_modified = FALSE;
Packit Service d328f3
	GHashTable *parameters = NULL;
Packit Service d328f3
	const char *p_name;
Packit Service d328f3
Packit Service d328f3
	g_assert (NM_IS_CONNECTION (connection));
Packit Service d328f3
Packit Service d328f3
	while ((p_name = va_arg (args, const char *))) {
Packit Service d328f3
		if (!parameters)
Packit Service d328f3
			parameters =  g_hash_table_new (g_str_hash, g_str_equal);
Packit Service d328f3
		g_hash_table_insert (parameters, (gpointer *) p_name, va_arg (args, gpointer));
Packit Service d328f3
	}
Packit Service d328f3
Packit Service d328f3
	success = nm_connection_normalize (connection,
Packit Service d328f3
	                                   parameters,
Packit Service d328f3
	                                   &was_modified,
Packit Service d328f3
	                                   &error);
Packit Service d328f3
	g_assert_no_error (error);
Packit Service d328f3
	g_assert (success);
Packit Service d328f3
Packit Service d328f3
	if (parameters)
Packit Service d328f3
		g_hash_table_destroy (parameters);
Packit Service d328f3
Packit Service d328f3
	return was_modified;
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static inline gboolean
Packit Service d328f3
_nmtst_connection_normalize (NMConnection *connection, ...)
Packit Service d328f3
{
Packit Service d328f3
	gboolean was_modified;
Packit Service d328f3
	va_list args;
Packit Service d328f3
Packit Service d328f3
	va_start (args, connection);
Packit Service d328f3
	was_modified = _nmtst_connection_normalize_v (connection, args);
Packit Service d328f3
	va_end (args);
Packit Service d328f3
Packit Service d328f3
	return was_modified;
Packit Service d328f3
}
Packit Service d328f3
#define nmtst_connection_normalize(connection, ...) \
Packit Service d328f3
    _nmtst_connection_normalize(connection, ##__VA_ARGS__, NULL)
Packit Service d328f3
Packit Service d328f3
static inline NMConnection *
Packit Service d328f3
_nmtst_connection_duplicate_and_normalize (NMConnection *connection, ...)
Packit Service d328f3
{
Packit Service d328f3
	va_list args;
Packit Service d328f3
Packit Service d328f3
	connection = nmtst_clone_connection (connection);
Packit Service d328f3
Packit Service d328f3
	va_start (args, connection);
Packit Service d328f3
	_nmtst_connection_normalize_v (connection, args);
Packit Service d328f3
	va_end (args);
Packit Service d328f3
Packit Service d328f3
	return connection;
Packit Service d328f3
}
Packit Service d328f3
#define nmtst_connection_duplicate_and_normalize(connection, ...) \
Packit Service d328f3
    _nmtst_connection_duplicate_and_normalize(connection, ##__VA_ARGS__, NULL)
Packit Service d328f3
Packit Service d328f3
static inline void
Packit Service d328f3
nmtst_assert_connection_equals (NMConnection *a, gboolean normalize_a, NMConnection *b, gboolean normalize_b)
Packit Service d328f3
{
Packit Service d328f3
	gboolean compare;
Packit Service d328f3
	gs_unref_object NMConnection *a2 = NULL;
Packit Service d328f3
	gs_unref_object NMConnection *b2 = NULL;
Packit Service d328f3
	GHashTable *out_settings = NULL;
Packit Service d328f3
Packit Service d328f3
	g_assert (NM_IS_CONNECTION (a));
Packit Service d328f3
	g_assert (NM_IS_CONNECTION (b));
Packit Service d328f3
Packit Service d328f3
	if (normalize_a)
Packit Service d328f3
		a = a2 = nmtst_connection_duplicate_and_normalize (a);
Packit Service d328f3
	if (normalize_b)
Packit Service d328f3
		b = b2 = nmtst_connection_duplicate_and_normalize (b);
Packit Service d328f3
Packit Service d328f3
	compare = nm_connection_diff (a, b, NM_SETTING_COMPARE_FLAG_EXACT, &out_settings);
Packit Service d328f3
	if (!compare || out_settings) {
Packit Service d328f3
		const char *name, *pname;
Packit Service d328f3
		GHashTable *setting;
Packit Service d328f3
		GHashTableIter iter, iter2;
Packit Service d328f3
Packit Service d328f3
		__NMTST_LOG (g_message, ">>> ASSERTION nmtst_assert_connection_equals() fails");
Packit Service d328f3
		if (out_settings) {
Packit Service d328f3
			g_hash_table_iter_init (&iter, out_settings);
Packit Service d328f3
			while (g_hash_table_iter_next (&iter, (gpointer *) &name, (gpointer *) &setting)) {
Packit Service d328f3
				__NMTST_LOG (g_message, ">>> differences in setting '%s':", name);
Packit Service d328f3
Packit Service d328f3
				g_hash_table_iter_init (&iter2, setting);
Packit Service d328f3
				while (g_hash_table_iter_next (&iter2, (gpointer *) &pname, NULL))
Packit Service d328f3
					__NMTST_LOG (g_message, ">>> differences in setting '%s.%s'", name, pname);
Packit Service d328f3
			}
Packit Service d328f3
		}
Packit Service d328f3
Packit Service d328f3
#ifdef __NM_KEYFILE_INTERNAL_H__
Packit Service d328f3
		{
Packit Service d328f3
			gs_unref_keyfile GKeyFile *kf_a = NULL, *kf_b = NULL;
Packit Service d328f3
			gs_free char *str_a = NULL, *str_b = NULL;
Packit Service d328f3
Packit Service d328f3
			kf_a = nm_keyfile_write (a, NULL, NULL, NULL);
Packit Service d328f3
			kf_b = nm_keyfile_write (b, NULL, NULL, NULL);
Packit Service d328f3
Packit Service d328f3
			if (kf_a)
Packit Service d328f3
				str_a = g_key_file_to_data (kf_a, NULL, NULL);
Packit Service d328f3
			if (kf_b)
Packit Service d328f3
				str_b = g_key_file_to_data (kf_b, NULL, NULL);
Packit Service d328f3
Packit Service d328f3
			__NMTST_LOG (g_message, ">>> Connection A as kf (*WARNING: keyfile representation might not show the difference*):\n%s", str_a);
Packit Service d328f3
			__NMTST_LOG (g_message, ">>> Connection B as kf (*WARNING: keyfile representation might not show the difference*):\n%s", str_b);
Packit Service d328f3
		}
Packit Service d328f3
#endif
Packit Service d328f3
	}
Packit Service d328f3
	g_assert (compare);
Packit Service d328f3
	g_assert (!out_settings);
Packit Service d328f3
Packit Service d328f3
	compare = nm_connection_compare (a, b, NM_SETTING_COMPARE_FLAG_EXACT);
Packit Service d328f3
	g_assert (compare);
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static inline void
Packit Service d328f3
nmtst_assert_connection_verifies (NMConnection *con)
Packit Service d328f3
{
Packit Service d328f3
	/* assert that the connection does verify, it might be normaliziable or not */
Packit Service d328f3
	GError *error = NULL;
Packit Service d328f3
	gboolean success;
Packit Service d328f3
Packit Service d328f3
	g_assert (NM_IS_CONNECTION (con));
Packit Service d328f3
Packit Service d328f3
	success = nm_connection_verify (con, &error);
Packit Service d328f3
	g_assert_no_error (error);
Packit Service d328f3
	g_assert (success);
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static inline void
Packit Service d328f3
nmtst_assert_connection_verifies_without_normalization (NMConnection *con)
Packit Service d328f3
{
Packit Service d328f3
	/* assert that the connection verifies and does not need any normalization */
Packit Service d328f3
	GError *error = NULL;
Packit Service d328f3
	gboolean success;
Packit Service d328f3
	gboolean was_modified = FALSE;
Packit Service d328f3
	gs_unref_object NMConnection *clone = NULL;
Packit Service d328f3
Packit Service d328f3
	clone = nmtst_clone_connection (con);
Packit Service d328f3
Packit Service d328f3
	nmtst_assert_connection_verifies (con);
Packit Service d328f3
Packit Service d328f3
	success = nm_connection_normalize (clone, NULL, &was_modified, &error);
Packit Service d328f3
	g_assert_no_error (error);
Packit Service d328f3
	g_assert (success);
Packit Service d328f3
	nmtst_assert_connection_equals (con, FALSE, clone, FALSE);
Packit Service d328f3
	g_assert (!was_modified);
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static inline void
Packit Service d328f3
nmtst_assert_connection_verifies_and_normalizable (NMConnection *con)
Packit Service d328f3
{
Packit Service d328f3
	/* assert that the connection does verify, but normalization still modifies it */
Packit Service d328f3
	GError *error = NULL;
Packit Service d328f3
	gboolean success;
Packit Service d328f3
	gboolean was_modified = FALSE;
Packit Service d328f3
	gs_unref_object NMConnection *clone = NULL;
Packit Service d328f3
Packit Service d328f3
	clone = nmtst_clone_connection (con);
Packit Service d328f3
Packit Service d328f3
	nmtst_assert_connection_verifies (con);
Packit Service d328f3
Packit Service d328f3
	success = nm_connection_normalize (clone, NULL, &was_modified, &error);
Packit Service d328f3
	g_assert_no_error (error);
Packit Service d328f3
	g_assert (success);
Packit Service d328f3
	g_assert (was_modified);
Packit Service d328f3
Packit Service d328f3
	/* again! */
Packit Service d328f3
	nmtst_assert_connection_verifies_without_normalization (clone);
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static inline void
Packit Service d328f3
nmtst_assert_connection_verifies_after_normalization (NMConnection *con,
Packit Service d328f3
                                                      GQuark expect_error_domain,
Packit Service d328f3
                                                      gint expect_error_code)
Packit Service d328f3
{
Packit Service d328f3
	/* assert that the connection does not verify, but normalization does fix it */
Packit Service d328f3
	GError *error = NULL;
Packit Service d328f3
	gboolean success;
Packit Service d328f3
	gboolean was_modified = FALSE;
Packit Service d328f3
	gs_unref_object NMConnection *clone = NULL;
Packit Service d328f3
Packit Service d328f3
	clone = nmtst_clone_connection (con);
Packit Service d328f3
Packit Service d328f3
	success = nm_connection_verify (con, &error);
Packit Service d328f3
	nmtst_assert_error (error, expect_error_domain, expect_error_code, NULL);
Packit Service d328f3
	g_assert (!success);
Packit Service d328f3
	g_clear_error (&error);
Packit Service d328f3
Packit Service d328f3
	success = nm_connection_normalize (clone, NULL, &was_modified, &error);
Packit Service d328f3
	g_assert_no_error (error);
Packit Service d328f3
	g_assert (success);
Packit Service d328f3
	g_assert (was_modified);
Packit Service d328f3
Packit Service d328f3
	/* again! */
Packit Service d328f3
	nmtst_assert_connection_verifies_without_normalization (clone);
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static inline void
Packit Service d328f3
nmtst_assert_connection_unnormalizable (NMConnection *con,
Packit Service d328f3
                                        GQuark expect_error_domain,
Packit Service d328f3
                                        gint expect_error_code)
Packit Service d328f3
{
Packit Service d328f3
	/* assert that the connection does not verify, and it cannot be fixed by normalization */
Packit Service d328f3
Packit Service d328f3
	GError *error = NULL;
Packit Service d328f3
	gboolean success;
Packit Service d328f3
	gboolean was_modified = FALSE;
Packit Service d328f3
	gs_unref_object NMConnection *clone = NULL;
Packit Service d328f3
Packit Service d328f3
	clone = nmtst_clone_connection (con);
Packit Service d328f3
Packit Service d328f3
	success = nm_connection_verify (con, &error);
Packit Service d328f3
	nmtst_assert_error (error, expect_error_domain, expect_error_code, NULL);
Packit Service d328f3
	g_assert (!success);
Packit Service d328f3
	g_clear_error (&error);
Packit Service d328f3
Packit Service d328f3
	success = nm_connection_normalize (clone, NULL, &was_modified, &error);
Packit Service d328f3
	nmtst_assert_error (error, expect_error_domain, expect_error_code, NULL);
Packit Service d328f3
	g_assert (!success);
Packit Service d328f3
	g_assert (!was_modified);
Packit Service d328f3
	nmtst_assert_connection_equals (con, FALSE, clone, FALSE);
Packit Service d328f3
	g_clear_error (&error);
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
static inline void
Packit Service d328f3
nmtst_assert_setting_verifies (NMSetting *setting)
Packit Service d328f3
{
Packit Service d328f3
	/* assert that the setting verifies without an error */
Packit Service d328f3
Packit Service d328f3
	GError *error = NULL;
Packit Service d328f3
	gboolean success;
Packit Service d328f3
Packit Service d328f3
	g_assert (NM_IS_SETTING (setting));
Packit Service d328f3
Packit Service d328f3
	success = nm_setting_verify (setting, NULL, &error);
Packit Service d328f3
	g_assert_no_error (error);
Packit Service d328f3
	g_assert (success);
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
#if defined(__NM_SIMPLE_CONNECTION_H__) && NM_CHECK_VERSION (1, 10, 0) && (!defined (NM_VERSION_MAX_ALLOWED) || NM_VERSION_MAX_ALLOWED >= NM_VERSION_1_10)
Packit Service d328f3
static inline void
Packit Service d328f3
_nmtst_assert_connection_has_settings (NMConnection *connection, gboolean has_at_least, gboolean has_at_most, ...)
Packit Service d328f3
{
Packit Service d328f3
	gs_unref_hashtable GHashTable *names = NULL;
Packit Service d328f3
	gs_free NMSetting **settings = NULL;
Packit Service d328f3
	va_list ap;
Packit Service d328f3
	const char *name;
Packit Service d328f3
	guint i, len;
Packit Service d328f3
	gs_unref_ptrarray GPtrArray *names_arr = NULL;
Packit Service d328f3
Packit Service d328f3
	g_assert (NM_IS_CONNECTION (connection));
Packit Service d328f3
Packit Service d328f3
	names = g_hash_table_new (g_str_hash, g_str_equal);
Packit Service d328f3
	names_arr = g_ptr_array_new ();
Packit Service d328f3
Packit Service d328f3
	va_start (ap, has_at_most);
Packit Service d328f3
	while ((name = va_arg (ap, const char *))) {
Packit Service d328f3
		if (!nm_g_hash_table_add (names, (gpointer) name))
Packit Service d328f3
			g_assert_not_reached ();
Packit Service d328f3
		g_ptr_array_add (names_arr, (gpointer) name);
Packit Service d328f3
	}
Packit Service d328f3
	va_end (ap);
Packit Service d328f3
Packit Service d328f3
	g_ptr_array_add (names_arr, NULL);
Packit Service d328f3
Packit Service d328f3
	settings = nm_connection_get_settings (connection, &len;;
Packit Service d328f3
	for (i = 0; i < len; i++) {
Packit Service d328f3
		if (   !g_hash_table_remove (names, nm_setting_get_name (settings[i]))
Packit Service d328f3
		    && has_at_most) {
Packit Service d328f3
			g_error ("nmtst_assert_connection_has_settings(): has setting \"%s\" which is not expected",
Packit Service d328f3
			         nm_setting_get_name (settings[i]));
Packit Service d328f3
		}
Packit Service d328f3
	}
Packit Service d328f3
	if (   g_hash_table_size (names) > 0
Packit Service d328f3
	    && has_at_least) {
Packit Service d328f3
		gs_free char *expected_str = g_strjoinv (" ", (char **) names_arr->pdata);
Packit Service d328f3
		gs_free const char **settings_names = NULL;
Packit Service d328f3
		gs_free char *has_str = NULL;
Packit Service d328f3
Packit Service d328f3
		settings_names = g_new0 (const char *, len + 1);
Packit Service d328f3
		for (i = 0; i < len; i++)
Packit Service d328f3
			settings_names[i] = nm_setting_get_name (settings[i]);
Packit Service d328f3
		has_str = g_strjoinv (" ", (char **) settings_names);
Packit Service d328f3
Packit Service d328f3
		g_error ("nmtst_assert_connection_has_settings(): the setting lacks %u expected settings (expected: [%s] vs. has: [%s])",
Packit Service d328f3
		         g_hash_table_size (names),
Packit Service d328f3
		         expected_str,
Packit Service d328f3
		         has_str);
Packit Service d328f3
	}
Packit Service d328f3
}
Packit Service d328f3
#define nmtst_assert_connection_has_settings(connection, ...)          _nmtst_assert_connection_has_settings ((connection), TRUE,  TRUE,  __VA_ARGS__, NULL)
Packit Service d328f3
#define nmtst_assert_connection_has_settings_at_least(connection, ...) _nmtst_assert_connection_has_settings ((connection), TRUE,  FALSE, __VA_ARGS__, NULL)
Packit Service d328f3
#define nmtst_assert_connection_has_settings_at_most(connection, ...)  _nmtst_assert_connection_has_settings ((connection), FALSE, TRUE,  __VA_ARGS__, NULL)
Packit Service d328f3
#endif
Packit Service d328f3
Packit Service d328f3
static inline void
Packit Service d328f3
nmtst_assert_setting_verify_fails (NMSetting *setting,
Packit Service d328f3
                                   GQuark expect_error_domain,
Packit Service d328f3
                                   gint expect_error_code)
Packit Service d328f3
{
Packit Service d328f3
	/* assert that the setting verification fails */
Packit Service d328f3
Packit Service d328f3
	GError *error = NULL;
Packit Service d328f3
	gboolean success;
Packit Service d328f3
Packit Service d328f3
	g_assert (NM_IS_SETTING (setting));
Packit Service d328f3
Packit Service d328f3
	success = nm_setting_verify (setting, NULL, &error);
Packit Service d328f3
	nmtst_assert_error (error, expect_error_domain, expect_error_code, NULL);
Packit Service d328f3
	g_assert (!success);
Packit Service d328f3
	g_clear_error (&error);
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
#endif
Packit Service d328f3
Packit Service d328f3
#ifdef __NM_UTILS_H__
Packit Service d328f3
static inline void
Packit Service d328f3
nmtst_assert_hwaddr_equals (gconstpointer hwaddr1, gssize hwaddr1_len, const char *expected, const char *file, int line)
Packit Service d328f3
{
Packit Service d328f3
	guint8 buf2[NM_UTILS_HWADDR_LEN_MAX];
Packit Service d328f3
	gsize hwaddr2_len = 1;
Packit Service d328f3
	const char *p;
Packit Service d328f3
	gboolean success;
Packit Service d328f3
Packit Service d328f3
	g_assert (hwaddr1_len > 0 && hwaddr1_len <= NM_UTILS_HWADDR_LEN_MAX);
Packit Service d328f3
Packit Service d328f3
	g_assert (expected);
Packit Service d328f3
	for (p = expected; *p; p++) {
Packit Service d328f3
		if (*p == ':' || *p == '-')
Packit Service d328f3
			hwaddr2_len++;
Packit Service d328f3
	}
Packit Service d328f3
	g_assert (hwaddr2_len <= NM_UTILS_HWADDR_LEN_MAX);
Packit Service d328f3
	g_assert (nm_utils_hwaddr_aton (expected, buf2, hwaddr2_len));
Packit Service d328f3
Packit Service d328f3
	/* Manually check the entire hardware address instead of using
Packit Service d328f3
	 * nm_utils_hwaddr_matches() because that function doesn't compare
Packit Service d328f3
	 * entire InfiniBand addresses for various (legitimate) reasons.
Packit Service d328f3
	 */
Packit Service d328f3
	success = (hwaddr1_len == hwaddr2_len);
Packit Service d328f3
	if (success)
Packit Service d328f3
		success = !memcmp (hwaddr1, buf2, hwaddr1_len);
Packit Service d328f3
	if (!success) {
Packit Service d328f3
		g_error ("assert: %s:%d: hwaddr '%s' (%zd) expected, but got %s (%zd)",
Packit Service d328f3
		         file, line, expected, hwaddr2_len, nm_utils_hwaddr_ntoa (hwaddr1, hwaddr1_len), hwaddr1_len);
Packit Service d328f3
	}
Packit Service d328f3
}
Packit Service d328f3
#define nmtst_assert_hwaddr_equals(hwaddr1, hwaddr1_len, expected) \
Packit Service d328f3
    nmtst_assert_hwaddr_equals (hwaddr1, hwaddr1_len, expected, __FILE__, __LINE__)
Packit Service d328f3
#endif
Packit Service d328f3
Packit Service d328f3
Packit Service d328f3
#if defined(__NM_SIMPLE_CONNECTION_H__) && defined(__NM_SETTING_CONNECTION_H__) && defined(__NM_KEYFILE_INTERNAL_H__)
Packit Service d328f3
Packit Service d328f3
static inline NMConnection *
Packit Service d328f3
nmtst_create_connection_from_keyfile (const char *keyfile_str, const char *keyfile_name, const char *base_dir)
Packit Service d328f3
{
Packit Service d328f3
	GKeyFile *keyfile;
Packit Service d328f3
	GError *error = NULL;
Packit Service d328f3
	gboolean success;
Packit Service d328f3
	NMConnection *con;
Packit Service d328f3
Packit Service d328f3
	g_assert (keyfile_str);
Packit Service d328f3
Packit Service d328f3
	keyfile =  g_key_file_new ();
Packit Service d328f3
	success = g_key_file_load_from_data (keyfile, keyfile_str, strlen (keyfile_str), G_KEY_FILE_NONE, &error);
Packit Service d328f3
	g_assert_no_error (error);
Packit Service d328f3
	g_assert (success);
Packit Service d328f3
Packit Service d328f3
	con = nm_keyfile_read (keyfile, keyfile_name, base_dir, NULL, NULL, &error);
Packit Service d328f3
	g_assert_no_error (error);
Packit Service d328f3
	g_assert (NM_IS_CONNECTION (con));
Packit Service d328f3
Packit Service d328f3
	g_key_file_unref (keyfile);
Packit Service d328f3
Packit Service d328f3
	nmtst_connection_normalize (con);
Packit Service d328f3
Packit Service d328f3
	return con;
Packit Service d328f3
}
Packit Service d328f3
Packit Service d328f3
#endif
Packit Service d328f3
Packit Service d328f3
#ifdef __NM_CONNECTION_H__
Packit Service d328f3
Packit Service d328f3
static inline GVariant *
Packit Service d328f3
_nmtst_variant_new_vardict (int dummy, ...)
Packit Service d328f3
{
Packit Service d328f3
	GVariantBuilder builder;
Packit Service d328f3
	va_list ap;
Packit Service d328f3
	const char *name;
Packit Service d328f3
	GVariant *variant;
Packit Service d328f3
Packit Service d328f3
	g_variant_builder_init (&builder, G_VARIANT_TYPE_VARDICT);
Packit Service d328f3
Packit Service d328f3
	va_start (ap, dummy);
Packit Service d328f3
	while ((name = va_arg (ap, const char *))) {
Packit Service d328f3
		variant = va_arg (ap, GVariant *);
Packit Service d328f3
		g_variant_builder_add (&builder, "{sv}", name, variant);
Packit Service d328f3
	}
Packit Service d328f3
	va_end (ap);
Packit Service d328f3
Packit Service d328f3
	return g_variant_builder_end (&builder);
Packit Service d328f3
}
Packit Service d328f3
#define nmtst_variant_new_vardict(...) _nmtst_variant_new_vardict (0, __VA_ARGS__, NULL)
Packit Service d328f3
Packit Service d328f3
#define nmtst_assert_variant_is_of_type(variant, type) \
Packit Service d328f3
	G_STMT_START { \
Packit Service d328f3
		GVariant *_variantx = (variant); \
Packit Service d328f3
		\
Packit Service d328f3
		g_assert (_variantx); \
Packit Service d328f3
		g_assert (g_variant_is_of_type (_variantx, (type))); \
Packit Service d328f3
	} G_STMT_END
Packit Service d328f3
Packit Service d328f3
#define nmtst_assert_variant_uint32(variant, val) \
Packit Service d328f3
	G_STMT_START { \
Packit Service d328f3
		GVariant *_variant = (variant); \
Packit Service d328f3
		\
Packit Service d328f3
		nmtst_assert_variant_is_of_type (_variant, G_VARIANT_TYPE_UINT32); \
Packit Service d328f3
		g_assert_cmpint (g_variant_get_uint32 (_variant), ==, (val)); \
Packit Service d328f3
	} G_STMT_END
Packit Service d328f3
Packit Service d328f3
#define nmtst_assert_variant_string(variant, str) \
Packit Service d328f3
	G_STMT_START { \
Packit Service d328f3
		gsize _l; \
Packit Service d328f3
		GVariant *_variant = (variant); \
Packit Service d328f3
		const char *_str = (str); \
Packit Service d328f3
		\
Packit Service d328f3
		nmtst_assert_variant_is_of_type (_variant, G_VARIANT_TYPE_STRING); \
Packit Service d328f3
		g_assert (_str); \
Packit Service d328f3
		g_assert_cmpstr (g_variant_get_string (_variant, &_l), ==, _str); \
Packit Service d328f3
		g_assert_cmpint (_l, ==, strlen (_str)); \
Packit Service d328f3
	} G_STMT_END
Packit Service d328f3
Packit Service d328f3
typedef enum {
Packit Service d328f3
	NMTST_VARIANT_EDITOR_CONNECTION,
Packit Service d328f3
	NMTST_VARIANT_EDITOR_SETTING,
Packit Service d328f3
	NMTST_VARIANT_EDITOR_PROPERTY
Packit Service d328f3
} NmtstVariantEditorPhase;
Packit Service d328f3
Packit Service d328f3
#define NMTST_VARIANT_EDITOR(__connection_variant, __code) \
Packit Service d328f3
	G_STMT_START { \
Packit Service d328f3
		GVariantIter __connection_iter, *__setting_iter; \
Packit Service d328f3
		GVariantBuilder __connection_builder, __setting_builder; \
Packit Service d328f3
		const char *__cur_setting_name, *__cur_property_name; \
Packit Service d328f3
		GVariant *__property_val; \
Packit Service d328f3
		NmtstVariantEditorPhase __phase; \
Packit Service d328f3
                                                                        \
Packit Service d328f3
		g_variant_builder_init (&__connection_builder, NM_VARIANT_TYPE_CONNECTION); \
Packit Service d328f3
		g_variant_iter_init (&__connection_iter, __connection_variant); \
Packit Service d328f3
		 \
Packit Service d328f3
		__phase = NMTST_VARIANT_EDITOR_CONNECTION; \
Packit Service d328f3
		__cur_setting_name = NULL; \
Packit Service d328f3
		__cur_property_name = NULL; \
Packit Service d328f3
		__code; \
Packit Service d328f3
		while (g_variant_iter_next (&__connection_iter, "{&sa{sv}}", &__cur_setting_name, &__setting_iter)) { \
Packit Service d328f3
			g_variant_builder_init (&__setting_builder, NM_VARIANT_TYPE_SETTING); \
Packit Service d328f3
			__phase = NMTST_VARIANT_EDITOR_SETTING; \
Packit Service d328f3
			__cur_property_name = NULL; \
Packit Service d328f3
			__code; \
Packit Service d328f3
			 \
Packit Service d328f3
			while (   __cur_setting_name \
Packit Service d328f3
			       && g_variant_iter_next (__setting_iter, "{&sv}", &__cur_property_name, &__property_val)) { \
Packit Service d328f3
				__phase = NMTST_VARIANT_EDITOR_PROPERTY; \
Packit Service d328f3
				__code; \
Packit Service d328f3
				 \
Packit Service d328f3
				if (__cur_property_name) { \
Packit Service d328f3
					g_variant_builder_add (&__setting_builder, "{sv}", \
Packit Service d328f3
					                       __cur_property_name, \
Packit Service d328f3
					                       __property_val); \
Packit Service d328f3
				} \
Packit Service d328f3
				g_variant_unref (__property_val); \
Packit Service d328f3
			} \
Packit Service d328f3
			 \
Packit Service d328f3
			if (__cur_setting_name) \
Packit Service d328f3
				g_variant_builder_add (&__connection_builder, "{sa{sv}}", __cur_setting_name, &__setting_builder); \
Packit Service d328f3
			else \
Packit Service d328f3
				g_variant_builder_clear (&__setting_builder); \
Packit Service d328f3
			g_variant_iter_free (__setting_iter); \
Packit Service d328f3
		} \
Packit Service d328f3
		 \
Packit Service d328f3
		g_variant_unref (__connection_variant); \
Packit Service d328f3
		 \
Packit Service d328f3
		__connection_variant = g_variant_builder_end (&__connection_builder); \
Packit Service d328f3
	} G_STMT_END;
Packit Service d328f3
Packit Service d328f3
#define NMTST_VARIANT_ADD_SETTING(__setting_name, __setting_variant) \
Packit Service d328f3
	G_STMT_START { \
Packit Service d328f3
		if (__phase == NMTST_VARIANT_EDITOR_CONNECTION) \
Packit Service d328f3
			g_variant_builder_add (&__connection_builder, "{s@a{sv}}", __setting_name, __setting_variant); \
Packit Service d328f3
	} G_STMT_END
Packit Service d328f3
Packit Service d328f3
#define NMTST_VARIANT_DROP_SETTING(__setting_name) \
Packit Service d328f3
	G_STMT_START { \
Packit Service d328f3
		if (__phase == NMTST_VARIANT_EDITOR_SETTING && __cur_setting_name) { \
Packit Service d328f3
			if (!strcmp (__cur_setting_name, __setting_name)) \
Packit Service d328f3
				__cur_setting_name = NULL; \
Packit Service d328f3
		} \
Packit Service d328f3
	} G_STMT_END
Packit Service d328f3
Packit Service d328f3
#define NMTST_VARIANT_ADD_PROPERTY(__setting_name, __property_name, __format_string, __value) \
Packit Service d328f3
	G_STMT_START { \
Packit Service d328f3
		if (__phase == NMTST_VARIANT_EDITOR_SETTING) { \
Packit Service d328f3
			if (!strcmp (__cur_setting_name, __setting_name)) { \
Packit Service d328f3
				g_variant_builder_add (&__setting_builder, "{sv}", __property_name, \
Packit Service d328f3
				                       g_variant_new (__format_string, __value)); \
Packit Service d328f3
			} \
Packit Service d328f3
		} \
Packit Service d328f3
	} G_STMT_END
Packit Service d328f3
Packit Service d328f3
#define NMTST_VARIANT_DROP_PROPERTY(__setting_name, __property_name) \
Packit Service d328f3
	G_STMT_START { \
Packit Service d328f3
		if (__phase == NMTST_VARIANT_EDITOR_PROPERTY && __cur_property_name) { \
Packit Service d328f3
			if (   !strcmp (__cur_setting_name, __setting_name) \
Packit Service d328f3
			    && !strcmp (__cur_property_name, __property_name)) \
Packit Service d328f3
				__cur_property_name = NULL; \
Packit Service d328f3
		} \
Packit Service d328f3
	} G_STMT_END
Packit Service d328f3
Packit Service d328f3
#define NMTST_VARIANT_CHANGE_PROPERTY(__setting_name, __property_name, __format_string, __value) \
Packit Service d328f3
	G_STMT_START { \
Packit Service d328f3
		NMTST_VARIANT_DROP_PROPERTY (__setting_name, __property_name); \
Packit Service d328f3
		NMTST_VARIANT_ADD_PROPERTY (__setting_name, __property_name, __format_string, __value); \
Packit Service d328f3
	} G_STMT_END
Packit Service d328f3
Packit Service d328f3
#endif /* __NM_CONNECTION_H__ */
Packit Service d328f3
Packit Service d328f3
#endif /* __NM_TEST_UTILS_H__ */