blob: 28a3684874089fb656db2a31fe06b94432e41759 [file] [log] [blame]
/* Utility functions for tests that rely on GLib
*
* Copyright © 2010-2011 Nokia Corporation
* Copyright © 2013-2015 Collabora Ltd.
*
* Permission is hereby granted, free of charge, to any person
* obtaining a copy of this software and associated documentation files
* (the "Software"), to deal in the Software without restriction,
* including without limitation the rights to use, copy, modify, merge,
* publish, distribute, sublicense, and/or sell copies of the Software,
* and to permit persons to whom the Software is furnished to do so,
* subject to the following conditions:
*
* The above copyright notice and this permission notice shall be
* included in all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
* BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
* ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE.
*/
#include <config.h>
#include "test-utils-glib.h"
#include <errno.h>
#include <string.h>
#ifdef DBUS_WIN
# include <io.h>
# include <windows.h>
#else
# include <netdb.h>
# include <signal.h>
# include <unistd.h>
# include <sys/socket.h>
# include <sys/types.h>
# include <pwd.h>
#endif
#include <glib.h>
#include <glib/gstdio.h>
#include <dbus/dbus.h>
#ifdef G_OS_WIN
# define isatty(x) _isatty(x)
#endif
void
_test_assert_no_error (const DBusError *e,
const char *file,
int line)
{
if (G_UNLIKELY (dbus_error_is_set (e)))
g_error ("%s:%d: expected success but got error: %s: %s",
file, line, e->name, e->message);
}
#ifdef DBUS_UNIX
static void
child_setup (gpointer user_data)
{
const struct passwd *pwd = user_data;
uid_t uid = geteuid ();
if (pwd == NULL || (pwd->pw_uid == uid && getuid () == uid))
return;
if (uid != 0)
g_error ("not currently euid 0: %lu", (unsigned long) uid);
if (setuid (pwd->pw_uid) != 0)
g_error ("could not setuid (%lu): %s",
(unsigned long) pwd->pw_uid, g_strerror (errno));
uid = getuid ();
if (uid != pwd->pw_uid)
g_error ("after successful setuid (%lu) my uid is %ld",
(unsigned long) pwd->pw_uid, (unsigned long) uid);
uid = geteuid ();
if (uid != pwd->pw_uid)
g_error ("after successful setuid (%lu) my euid is %ld",
(unsigned long) pwd->pw_uid, (unsigned long) uid);
}
#endif
static gchar *
spawn_dbus_daemon (const gchar *binary,
const gchar *configuration,
const gchar *listen_address,
TestUser user,
const gchar *runtime_dir,
GPid *daemon_pid)
{
GError *error = NULL;
GString *address;
gint address_fd;
GPtrArray *argv;
gchar **envp;
#ifdef DBUS_UNIX
const struct passwd *pwd = NULL;
#endif
if (user != TEST_USER_ME)
{
#ifdef DBUS_UNIX
if (getuid () != 0)
{
g_test_skip ("cannot use alternative uid when not uid 0");
return NULL;
}
switch (user)
{
case TEST_USER_ROOT:
break;
case TEST_USER_MESSAGEBUS:
pwd = getpwnam (DBUS_USER);
if (pwd == NULL)
{
gchar *message = g_strdup_printf ("user '%s' does not exist",
DBUS_USER);
g_test_skip (message);
g_free (message);
return NULL;
}
break;
case TEST_USER_OTHER:
pwd = getpwnam (DBUS_TEST_USER);
if (pwd == NULL)
{
gchar *message = g_strdup_printf ("user '%s' does not exist",
DBUS_TEST_USER);
g_test_skip (message);
g_free (message);
return NULL;
}
break;
case TEST_USER_ME:
/* cannot get here, fall through */
default:
g_assert_not_reached ();
}
#else
g_test_skip ("cannot use alternative uid on Windows");
return NULL;
#endif
}
envp = g_get_environ ();
if (runtime_dir != NULL)
envp = g_environ_setenv (envp, "XDG_RUNTIME_DIR", runtime_dir, TRUE);
argv = g_ptr_array_new_with_free_func (g_free);
g_ptr_array_add (argv, g_strdup (binary));
g_ptr_array_add (argv, g_strdup (configuration));
g_ptr_array_add (argv, g_strdup ("--nofork"));
g_ptr_array_add (argv, g_strdup ("--print-address=1")); /* stdout */
if (listen_address != NULL)
g_ptr_array_add (argv, g_strdup (listen_address));
#ifdef DBUS_UNIX
g_ptr_array_add (argv, g_strdup ("--systemd-activation"));
#endif
g_ptr_array_add (argv, NULL);
g_spawn_async_with_pipes (NULL, /* working directory */
(gchar **) argv->pdata,
envp,
G_SPAWN_DO_NOT_REAP_CHILD | G_SPAWN_SEARCH_PATH,
#ifdef DBUS_UNIX
child_setup, (gpointer) pwd,
#else
NULL, NULL,
#endif
daemon_pid,
NULL, /* child's stdin = /dev/null */
&address_fd,
NULL, /* child's stderr = our stderr */
&error);
g_assert_no_error (error);
g_ptr_array_free (argv, TRUE);
g_strfreev (envp);
address = g_string_new (NULL);
/* polling until the dbus-daemon writes out its address is a bit stupid,
* but at least it's simple, unlike dbus-launch... in principle we could
* use select() here, but life's too short */
while (1)
{
gssize bytes;
gchar buf[4096];
gchar *newline;
bytes = read (address_fd, buf, sizeof (buf));
if (bytes > 0)
g_string_append_len (address, buf, bytes);
newline = strchr (address->str, '\n');
if (newline != NULL)
{
if ((newline > address->str) && ('\r' == newline[-1]))
newline -= 1;
g_string_truncate (address, newline - address->str);
break;
}
g_usleep (G_USEC_PER_SEC / 10);
}
g_close (address_fd, NULL);
return g_string_free (address, FALSE);
}
gchar *
test_get_dbus_daemon (const gchar *config_file,
TestUser user,
const gchar *runtime_dir,
GPid *daemon_pid)
{
gchar *dbus_daemon;
gchar *arg;
const gchar *listen_address = NULL;
gchar *address;
/* we often have to override this because on Windows, the default may be
* autolaunch:, which is globally-scoped and hence unsuitable for
* regression tests */
listen_address = "--address=" TEST_LISTEN;
if (config_file != NULL)
{
if (g_getenv ("DBUS_TEST_DATA") == NULL)
{
g_test_message ("set DBUS_TEST_DATA to a directory containing %s",
config_file);
g_test_skip ("DBUS_TEST_DATA not set");
return NULL;
}
arg = g_strdup_printf (
"--config-file=%s/%s",
g_getenv ("DBUS_TEST_DATA"), config_file);
/* The configuration file is expected to give a suitable address,
* do not override it */
listen_address = NULL;
}
else if (g_getenv ("DBUS_TEST_DATADIR") != NULL)
{
arg = g_strdup_printf ("--config-file=%s/dbus-1/session.conf",
g_getenv ("DBUS_TEST_DATADIR"));
}
else if (g_getenv ("DBUS_TEST_DATA") != NULL)
{
arg = g_strdup_printf (
"--config-file=%s/valid-config-files/session.conf",
g_getenv ("DBUS_TEST_DATA"));
}
else
{
arg = g_strdup ("--session");
}
dbus_daemon = g_strdup (g_getenv ("DBUS_TEST_DAEMON"));
if (dbus_daemon == NULL)
dbus_daemon = g_strdup ("dbus-daemon");
if (g_getenv ("DBUS_TEST_DAEMON_ADDRESS") != NULL)
{
if (config_file != NULL || user != TEST_USER_ME)
{
g_test_skip ("cannot use DBUS_TEST_DAEMON_ADDRESS for "
"unusally-configured dbus-daemon");
address = NULL;
}
else
{
address = g_strdup (g_getenv ("DBUS_TEST_DAEMON_ADDRESS"));
}
}
else
{
address = spawn_dbus_daemon (dbus_daemon, arg,
listen_address, user, runtime_dir, daemon_pid);
}
g_free (dbus_daemon);
g_free (arg);
return address;
}
DBusConnection *
test_connect_to_bus (TestMainContext *ctx,
const gchar *address)
{
DBusConnection *conn;
DBusError error = DBUS_ERROR_INIT;
dbus_bool_t ok;
conn = dbus_connection_open_private (address, &error);
test_assert_no_error (&error);
g_assert (conn != NULL);
ok = dbus_bus_register (conn, &error);
test_assert_no_error (&error);
g_assert (ok);
g_assert (dbus_bus_get_unique_name (conn) != NULL);
if (ctx != NULL)
test_connection_setup (ctx, conn);
return conn;
}
DBusConnection *
test_connect_to_bus_as_user (TestMainContext *ctx,
const char *address,
TestUser user)
{
/* For now we only do tests like this on Linux, because I don't know how
* safe this use of setresuid() is on other platforms */
#if defined(HAVE_GETRESUID) && defined(HAVE_SETRESUID) && defined(__linux__)
uid_t ruid, euid, suid;
const struct passwd *pwd;
DBusConnection *conn;
const char *username;
switch (user)
{
case TEST_USER_ME:
return test_connect_to_bus (ctx, address);
case TEST_USER_ROOT:
username = "root";
break;
case TEST_USER_MESSAGEBUS:
username = DBUS_USER;
break;
case TEST_USER_OTHER:
username = DBUS_TEST_USER;
break;
default:
g_return_val_if_reached (NULL);
}
if (getresuid (&ruid, &euid, &suid) != 0)
g_error ("getresuid: %s", g_strerror (errno));
if (ruid != 0 || euid != 0 || suid != 0)
{
g_test_message ("not uid 0 (ruid=%ld euid=%ld suid=%ld)",
(unsigned long) ruid, (unsigned long) euid, (unsigned long) suid);
g_test_skip ("not uid 0");
return NULL;
}
pwd = getpwnam (username);
if (pwd == NULL)
{
g_test_message ("getpwnam(\"%s\"): %s", username, g_strerror (errno));
g_test_skip ("not uid 0");
return NULL;
}
/* Impersonate the desired user while we connect to the bus.
* This should work, because we're root. */
if (setresuid (pwd->pw_uid, pwd->pw_uid, 0) != 0)
g_error ("setresuid(%ld, (same), 0): %s",
(unsigned long) pwd->pw_uid, g_strerror (errno));
conn = test_connect_to_bus (ctx, address);
/* go back to our saved uid */
if (setresuid (0, 0, 0) != 0)
g_error ("setresuid(0, 0, 0): %s", g_strerror (errno));
return conn;
#else
switch (user)
{
case TEST_USER_ME:
return test_connect_to_bus (ctx, address);
case TEST_USER_ROOT:
case TEST_USER_MESSAGEBUS:
case TEST_USER_OTHER:
g_test_skip ("setresuid() not available, or unsure about "
"credentials-passing semantics on this platform");
return NULL;
default:
g_return_val_if_reached (NULL);
}
#endif
}
static void
pid_died (GPid pid,
gint status,
gpointer user_data)
{
gboolean *result = user_data;
g_assert (result != NULL);
g_assert (!*result);
*result = TRUE;
}
void
test_kill_pid (GPid pid)
{
gint died = FALSE;
g_child_watch_add (pid, pid_died, &died);
#ifdef DBUS_WIN
if (pid != NULL)
TerminateProcess (pid, 1);
#else
if (pid > 0)
kill (pid, SIGTERM);
#endif
while (!died)
g_main_context_iteration (NULL, TRUE);
}
static gboolean
time_out (gpointer data)
{
puts ("Bail out! Test timed out (GLib main loop timeout callback reached)");
fflush (stdout);
abort ();
return FALSE;
}
#ifdef G_OS_UNIX
static void wrap_abort (int signal) _DBUS_GNUC_NORETURN;
static void
wrap_abort (int signal)
{
/* We might be halfway through writing out something else, so force this
* onto its own line */
const char message [] = "\nBail out! Test timed out (SIGALRM received)\n";
if (write (STDOUT_FILENO, message, sizeof (message) - 1) <
(ssize_t) sizeof (message) - 1)
{
/* ignore short write - what would we do about it? */
}
abort ();
}
#endif
static void
set_timeout (guint factor)
{
static guint timeout = 0;
/* Prevent tests from hanging forever. This is intended to be long enough
* that any reasonable regression test on any reasonable hardware would
* have finished. */
#define TIMEOUT 60
if (timeout != 0)
g_source_remove (timeout);
timeout = g_timeout_add_seconds (TIMEOUT * factor, time_out, NULL);
#ifdef G_OS_UNIX
/* The GLib main loop might not be running (we don't use it in every
* test). Die with SIGALRM shortly after if necessary. */
alarm ((TIMEOUT * factor) + 10);
/* Get a log message and a core dump from the SIGALRM. */
{
struct sigaction act = { };
act.sa_handler = wrap_abort;
sigaction (SIGALRM, &act, NULL);
}
#endif
}
void
test_init (int *argcp, char ***argvp)
{
g_test_init (argcp, argvp, NULL);
g_test_bug_base ("https://bugs.freedesktop.org/show_bug.cgi?id=");
set_timeout (1);
}
static void
report_and_destroy (gpointer p)
{
GTimer *timer = p;
g_test_message ("Time since timeout reset %p: %.3f seconds",
timer, g_timer_elapsed (timer, NULL));
g_timer_destroy (timer);
}
void
test_timeout_reset (guint factor)
{
GTimer *timer = g_timer_new ();
g_test_message ("Resetting test timeout (reference: %p; factor: %u)",
timer, factor);
set_timeout (factor);
g_test_queue_destroy (report_and_destroy, timer);
}
void
test_progress (char symbol)
{
if (g_test_verbose () && isatty (1))
g_print ("%c", symbol);
}
/*
* Delete @path, with a retry loop if the system call is interrupted by
* an async signal. If @path does not exist, ignore; otherwise, it is
* required to be a non-directory.
*/
void
test_remove_if_exists (const gchar *path)
{
while (g_remove (path) != 0)
{
int saved_errno = errno;
if (saved_errno == ENOENT)
return;
#ifdef G_OS_UNIX
if (saved_errno == EINTR)
continue;
#endif
g_error ("Unable to remove file \"%s\": %s", path,
g_strerror (saved_errno));
}
}
/*
* Delete empty directory @path, with a retry loop if the system call is
* interrupted by an async signal. @path is required to exist.
*/
void
test_rmdir_must_exist (const gchar *path)
{
while (g_remove (path) != 0)
{
int saved_errno = errno;
#ifdef G_OS_UNIX
if (saved_errno == EINTR)
continue;
#endif
g_error ("Unable to remove directory \"%s\": %s", path,
g_strerror (saved_errno));
}
}
/*
* Delete empty directory @path, with a retry loop if the system call is
* interrupted by an async signal. If @path does not exist, ignore.
*/
void
test_rmdir_if_exists (const gchar *path)
{
while (g_remove (path) != 0)
{
int saved_errno = errno;
if (saved_errno == ENOENT)
return;
#ifdef G_OS_UNIX
if (saved_errno == EINTR)
continue;
#endif
g_error ("Unable to remove directory \"%s\": %s", path,
g_strerror (saved_errno));
}
}
/*
* Create directory @path, with a retry loop if the system call is
* interrupted by an async signal.
*/
void
test_mkdir (const gchar *path,
gint mode)
{
while (g_mkdir (path, mode) != 0)
{
int saved_errno = errno;
#ifdef G_OS_UNIX
if (saved_errno == EINTR)
continue;
#endif
g_error ("Unable to create directory \"%s\": %s", path,
g_strerror (saved_errno));
}
}
gboolean
test_check_tcp_works (void)
{
#ifdef DBUS_UNIX
/* In pathological container environments, we might not have a
* working 127.0.0.1 */
int res;
struct addrinfo *addrs = NULL;
struct addrinfo hints;
int saved_errno;
_DBUS_ZERO (hints);
#ifdef AI_ADDRCONFIG
hints.ai_flags |= AI_ADDRCONFIG;
#endif
hints.ai_flags = AI_ADDRCONFIG;
hints.ai_family = AF_INET;
hints.ai_socktype = SOCK_STREAM;
hints.ai_protocol = IPPROTO_TCP;
res = getaddrinfo ("127.0.0.1", "0", &hints, &addrs);
saved_errno = errno;
if (res != 0)
{
const gchar *system_message;
gchar *skip_message;
#ifdef EAI_SYSTEM
if (res == EAI_SYSTEM)
system_message = g_strerror (saved_errno);
else
#endif
system_message = gai_strerror (res);
skip_message = g_strdup_printf ("Name resolution does not work here: "
"getaddrinfo(\"127.0.0.1\", \"0\", "
"{flags=ADDRCONFIG, family=INET,"
"socktype=STREAM, protocol=TCP}): "
"%s",
system_message);
g_test_skip (skip_message);
free (skip_message);
}
if (addrs != NULL)
freeaddrinfo (addrs);
return (res == 0);
#else
/* Assume that on Windows, TCP always works */
return TRUE;
#endif
}